debuginfo: Simplify TypeMap used during LLVM debuginfo generation.
The previous implementation was written before types were properly normalized for code generation and had to assume a more complicated relationship between types and their debuginfo -- generating separate identifiers for debuginfo nodes that were based on normalized types. Since types are now already normalized, we can use them as identifiers for debuginfo nodes.
This commit is contained in:
parent
026d8ce7f5
commit
e72e6399b1
6 changed files with 238 additions and 351 deletions
|
@ -3524,6 +3524,7 @@ dependencies = [
|
||||||
"rustc_hir",
|
"rustc_hir",
|
||||||
"rustc_index",
|
"rustc_index",
|
||||||
"rustc_llvm",
|
"rustc_llvm",
|
||||||
|
"rustc_macros",
|
||||||
"rustc_metadata",
|
"rustc_metadata",
|
||||||
"rustc_middle",
|
"rustc_middle",
|
||||||
"rustc_query_system",
|
"rustc_query_system",
|
||||||
|
|
|
@ -25,6 +25,7 @@ rustc_fs_util = { path = "../rustc_fs_util" }
|
||||||
rustc_hir = { path = "../rustc_hir" }
|
rustc_hir = { path = "../rustc_hir" }
|
||||||
rustc_index = { path = "../rustc_index" }
|
rustc_index = { path = "../rustc_index" }
|
||||||
rustc_llvm = { path = "../rustc_llvm" }
|
rustc_llvm = { path = "../rustc_llvm" }
|
||||||
|
rustc_macros = { path = "../rustc_macros" }
|
||||||
rustc_metadata = { path = "../rustc_metadata" }
|
rustc_metadata = { path = "../rustc_metadata" }
|
||||||
rustc_query_system = { path = "../rustc_query_system" }
|
rustc_query_system = { path = "../rustc_query_system" }
|
||||||
rustc_session = { path = "../rustc_session" }
|
rustc_session = { path = "../rustc_session" }
|
||||||
|
|
|
@ -23,9 +23,7 @@ use cstr::cstr;
|
||||||
use rustc_codegen_ssa::debuginfo::type_names::cpp_like_debuginfo;
|
use rustc_codegen_ssa::debuginfo::type_names::cpp_like_debuginfo;
|
||||||
use rustc_codegen_ssa::debuginfo::type_names::VTableNameKind;
|
use rustc_codegen_ssa::debuginfo::type_names::VTableNameKind;
|
||||||
use rustc_codegen_ssa::traits::*;
|
use rustc_codegen_ssa::traits::*;
|
||||||
use rustc_data_structures::fingerprint::Fingerprint;
|
|
||||||
use rustc_data_structures::fx::FxHashMap;
|
use rustc_data_structures::fx::FxHashMap;
|
||||||
use rustc_data_structures::stable_hasher::{HashStable, StableHasher};
|
|
||||||
use rustc_fs_util::path_to_c_string;
|
use rustc_fs_util::path_to_c_string;
|
||||||
use rustc_hir::def::CtorKind;
|
use rustc_hir::def::CtorKind;
|
||||||
use rustc_hir::def_id::{DefId, LOCAL_CRATE};
|
use rustc_hir::def_id::{DefId, LOCAL_CRATE};
|
||||||
|
@ -37,7 +35,6 @@ use rustc_middle::ty::subst::GenericArgKind;
|
||||||
use rustc_middle::ty::{
|
use rustc_middle::ty::{
|
||||||
self, AdtKind, GeneratorSubsts, Instance, ParamEnv, Ty, TyCtxt, COMMON_VTABLE_ENTRIES,
|
self, AdtKind, GeneratorSubsts, Instance, ParamEnv, Ty, TyCtxt, COMMON_VTABLE_ENTRIES,
|
||||||
};
|
};
|
||||||
use rustc_query_system::ich::NodeIdHashingMode;
|
|
||||||
use rustc_session::config::{self, DebugInfo};
|
use rustc_session::config::{self, DebugInfo};
|
||||||
use rustc_span::symbol::Symbol;
|
use rustc_span::symbol::Symbol;
|
||||||
use rustc_span::FileNameDisplayPreference;
|
use rustc_span::FileNameDisplayPreference;
|
||||||
|
@ -45,9 +42,11 @@ use rustc_span::{self, SourceFile, SourceFileHash};
|
||||||
use rustc_target::abi::{Abi, Align, HasDataLayout, Integer, TagEncoding};
|
use rustc_target::abi::{Abi, Align, HasDataLayout, Integer, TagEncoding};
|
||||||
use rustc_target::abi::{Int, Pointer, F32, F64};
|
use rustc_target::abi::{Int, Pointer, F32, F64};
|
||||||
use rustc_target::abi::{Primitive, Size, VariantIdx, Variants};
|
use rustc_target::abi::{Primitive, Size, VariantIdx, Variants};
|
||||||
|
use smallvec::SmallVec;
|
||||||
use tracing::debug;
|
use tracing::debug;
|
||||||
|
|
||||||
use libc::{c_longlong, c_uint};
|
use libc::{c_longlong, c_uint};
|
||||||
|
use std::cell::RefCell;
|
||||||
use std::collections::hash_map::Entry;
|
use std::collections::hash_map::Entry;
|
||||||
use std::fmt::{self, Write};
|
use std::fmt::{self, Write};
|
||||||
use std::hash::{Hash, Hasher};
|
use std::hash::{Hash, Hasher};
|
||||||
|
@ -95,193 +94,114 @@ pub const UNKNOWN_COLUMN_NUMBER: c_uint = 0;
|
||||||
pub const NO_SCOPE_METADATA: Option<&DIScope> = None;
|
pub const NO_SCOPE_METADATA: Option<&DIScope> = None;
|
||||||
|
|
||||||
mod unique_type_id {
|
mod unique_type_id {
|
||||||
use super::*;
|
use rustc_data_structures::{
|
||||||
use rustc_arena::DroplessArena;
|
fingerprint::Fingerprint,
|
||||||
|
stable_hasher::{HashStable, NodeIdHashingMode, StableHasher},
|
||||||
|
};
|
||||||
|
use rustc_middle::ty::{ParamEnv, PolyExistentialTraitRef, Ty, TyCtxt};
|
||||||
|
use rustc_target::abi::VariantIdx;
|
||||||
|
|
||||||
#[derive(Copy, Hash, Eq, PartialEq, Clone)]
|
// This type cannot be constructed outside of this module because
|
||||||
pub(super) struct UniqueTypeId(u32);
|
// it has a private field. We make use of this in order to prevent
|
||||||
|
// `UniqueTypeId` from being constructed directly, without asserting
|
||||||
// The `&'static str`s in this type actually point into the arena.
|
// the preconditions.
|
||||||
//
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, HashStable)]
|
||||||
// The `FxHashMap`+`Vec` pair could be replaced by `FxIndexSet`, but #75278
|
pub struct HiddenZst {
|
||||||
// found that to regress performance up to 2% in some cases. This might be
|
_inaccessible: (),
|
||||||
// revisited after further improvements to `indexmap`.
|
|
||||||
#[derive(Default)]
|
|
||||||
pub(super) struct TypeIdInterner {
|
|
||||||
arena: DroplessArena,
|
|
||||||
names: FxHashMap<&'static str, UniqueTypeId>,
|
|
||||||
strings: Vec<&'static str>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TypeIdInterner {
|
/// A unique identifier for anything that we create a debuginfo node for.
|
||||||
#[inline]
|
/// The types it contains are expected to already be normalized (which
|
||||||
pub(super) fn intern(&mut self, string: &str) -> UniqueTypeId {
|
/// is debug_asserted in the constructors).
|
||||||
if let Some(&name) = self.names.get(string) {
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, HashStable)]
|
||||||
return name;
|
pub(super) enum UniqueTypeId<'tcx> {
|
||||||
}
|
Ty(Ty<'tcx>, HiddenZst),
|
||||||
|
Variant(Ty<'tcx>, VariantIdx, HiddenZst),
|
||||||
|
VariantPart(Ty<'tcx>, HiddenZst),
|
||||||
|
VTableTy(Ty<'tcx>, Option<PolyExistentialTraitRef<'tcx>>, HiddenZst),
|
||||||
|
}
|
||||||
|
|
||||||
let name = UniqueTypeId(self.strings.len() as u32);
|
impl<'tcx> UniqueTypeId<'tcx> {
|
||||||
|
pub fn for_ty(tcx: TyCtxt<'tcx>, t: Ty<'tcx>) -> Self {
|
||||||
// `from_utf8_unchecked` is safe since we just allocated a `&str` which is known to be
|
debug_assert_eq!(t, tcx.normalize_erasing_regions(ParamEnv::reveal_all(), t));
|
||||||
// UTF-8.
|
UniqueTypeId::Ty(t, HiddenZst { _inaccessible: () })
|
||||||
let string: &str =
|
|
||||||
unsafe { std::str::from_utf8_unchecked(self.arena.alloc_slice(string.as_bytes())) };
|
|
||||||
// It is safe to extend the arena allocation to `'static` because we only access
|
|
||||||
// these while the arena is still alive.
|
|
||||||
let string: &'static str = unsafe { &*(string as *const str) };
|
|
||||||
self.strings.push(string);
|
|
||||||
self.names.insert(string, name);
|
|
||||||
name
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the symbol as a string. `Symbol::as_str()` should be used in
|
pub fn for_enum_variant(
|
||||||
// preference to this function.
|
tcx: TyCtxt<'tcx>,
|
||||||
pub(super) fn get(&self, symbol: UniqueTypeId) -> &str {
|
enum_ty: Ty<'tcx>,
|
||||||
self.strings[symbol.0 as usize]
|
variant_idx: VariantIdx,
|
||||||
|
) -> Self {
|
||||||
|
debug_assert_eq!(
|
||||||
|
enum_ty,
|
||||||
|
tcx.normalize_erasing_regions(ParamEnv::reveal_all(), enum_ty)
|
||||||
|
);
|
||||||
|
UniqueTypeId::Variant(enum_ty, variant_idx, HiddenZst { _inaccessible: () })
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn for_enum_variant_part(tcx: TyCtxt<'tcx>, enum_ty: Ty<'tcx>) -> Self {
|
||||||
|
debug_assert_eq!(
|
||||||
|
enum_ty,
|
||||||
|
tcx.normalize_erasing_regions(ParamEnv::reveal_all(), enum_ty)
|
||||||
|
);
|
||||||
|
UniqueTypeId::VariantPart(enum_ty, HiddenZst { _inaccessible: () })
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn for_vtable_ty(
|
||||||
|
tcx: TyCtxt<'tcx>,
|
||||||
|
self_type: Ty<'tcx>,
|
||||||
|
implemented_trait: Option<PolyExistentialTraitRef<'tcx>>,
|
||||||
|
) -> Self {
|
||||||
|
debug_assert_eq!(
|
||||||
|
self_type,
|
||||||
|
tcx.normalize_erasing_regions(ParamEnv::reveal_all(), self_type)
|
||||||
|
);
|
||||||
|
debug_assert_eq!(
|
||||||
|
implemented_trait,
|
||||||
|
tcx.normalize_erasing_regions(ParamEnv::reveal_all(), implemented_trait)
|
||||||
|
);
|
||||||
|
UniqueTypeId::VTableTy(self_type, implemented_trait, HiddenZst { _inaccessible: () })
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn to_string(&self, tcx: TyCtxt<'tcx>) -> String {
|
||||||
|
let mut hasher = StableHasher::new();
|
||||||
|
let mut hcx = tcx.create_stable_hashing_context();
|
||||||
|
hcx.while_hashing_spans(false, |hcx| {
|
||||||
|
hcx.with_node_id_hashing_mode(NodeIdHashingMode::HashDefPath, |hcx| {
|
||||||
|
self.hash_stable(hcx, &mut hasher);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
hasher.finish::<Fingerprint>().to_hex()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
use unique_type_id::*;
|
use unique_type_id::*;
|
||||||
|
|
||||||
/// The `TypeMap` is where the `CrateDebugContext` holds the type metadata nodes
|
/// The `TypeMap` is where the debug context holds the type metadata nodes
|
||||||
/// created so far. The metadata nodes are indexed by `UniqueTypeId`, and, for
|
/// created so far. The metadata nodes are indexed by `UniqueTypeId`.
|
||||||
/// faster lookup, also by `Ty`. The `TypeMap` is responsible for creating
|
|
||||||
/// `UniqueTypeId`s.
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
pub struct TypeMap<'ll, 'tcx> {
|
pub struct TypeMap<'ll, 'tcx> {
|
||||||
/// The `UniqueTypeId`s created so far.
|
unique_id_to_metadata: RefCell<FxHashMap<UniqueTypeId<'tcx>, &'ll DIType>>,
|
||||||
unique_id_interner: TypeIdInterner,
|
|
||||||
/// A map from `UniqueTypeId` to debuginfo metadata for that type. This is a 1:1 mapping.
|
|
||||||
unique_id_to_metadata: FxHashMap<UniqueTypeId, &'ll DIType>,
|
|
||||||
/// A map from types to debuginfo metadata. This is an N:1 mapping.
|
|
||||||
type_to_metadata: FxHashMap<Ty<'tcx>, &'ll DIType>,
|
|
||||||
/// A map from types to `UniqueTypeId`. This is an N:1 mapping.
|
|
||||||
type_to_unique_id: FxHashMap<Ty<'tcx>, UniqueTypeId>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'ll, 'tcx> TypeMap<'ll, 'tcx> {
|
impl<'ll, 'tcx> TypeMap<'ll, 'tcx> {
|
||||||
/// Adds a Ty to metadata mapping to the TypeMap. The method will fail if
|
|
||||||
/// the mapping already exists.
|
|
||||||
fn register_type_with_metadata(&mut self, type_: Ty<'tcx>, metadata: &'ll DIType) {
|
|
||||||
if self.type_to_metadata.insert(type_, metadata).is_some() {
|
|
||||||
bug!("type metadata for `Ty` '{}' is already in the `TypeMap`!", type_);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Removes a `Ty`-to-metadata mapping.
|
|
||||||
/// This is useful when computing the metadata for a potentially
|
|
||||||
/// recursive type (e.g., a function pointer of the form:
|
|
||||||
///
|
|
||||||
/// fn foo() -> impl Copy { foo }
|
|
||||||
///
|
|
||||||
/// This kind of type cannot be properly represented
|
|
||||||
/// via LLVM debuginfo. As a workaround,
|
|
||||||
/// we register a temporary Ty to metadata mapping
|
|
||||||
/// for the function before we compute its actual metadata.
|
|
||||||
/// If the metadata computation ends up recursing back to the
|
|
||||||
/// original function, it will use the temporary mapping
|
|
||||||
/// for the inner self-reference, preventing us from
|
|
||||||
/// recursing forever.
|
|
||||||
///
|
|
||||||
/// This function is used to remove the temporary metadata
|
|
||||||
/// mapping after we've computed the actual metadata.
|
|
||||||
fn remove_type(&mut self, ty: Ty<'tcx>) {
|
|
||||||
if self.type_to_metadata.remove(&ty).is_none() {
|
|
||||||
bug!("type metadata `Ty` '{}' is not in the `TypeMap`!", ty);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Adds a `UniqueTypeId` to metadata mapping to the `TypeMap`. The method will
|
/// Adds a `UniqueTypeId` to metadata mapping to the `TypeMap`. The method will
|
||||||
/// fail if the mapping already exists.
|
/// fail if the mapping already exists.
|
||||||
fn register_unique_id_with_metadata(
|
fn register_unique_id_with_metadata(
|
||||||
&mut self,
|
&self,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
metadata: &'ll DIType,
|
metadata: &'ll DIType,
|
||||||
) {
|
) {
|
||||||
if self.unique_id_to_metadata.insert(unique_type_id, metadata).is_some() {
|
if self.unique_id_to_metadata.borrow_mut().insert(unique_type_id, metadata).is_some() {
|
||||||
bug!(
|
bug!("type metadata for unique ID '{:?}' is already in the `TypeMap`!", unique_type_id);
|
||||||
"type metadata for unique ID '{}' is already in the `TypeMap`!",
|
|
||||||
self.get_unique_type_id_as_string(unique_type_id)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn find_metadata_for_type(&self, type_: Ty<'tcx>) -> Option<&'ll DIType> {
|
fn find_metadata_for_unique_id(
|
||||||
self.type_to_metadata.get(&type_).cloned()
|
&self,
|
||||||
}
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
|
) -> Option<&'ll DIType> {
|
||||||
fn find_metadata_for_unique_id(&self, unique_type_id: UniqueTypeId) -> Option<&'ll DIType> {
|
self.unique_id_to_metadata.borrow().get(&unique_type_id).cloned()
|
||||||
self.unique_id_to_metadata.get(&unique_type_id).cloned()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Gets the string representation of a `UniqueTypeId`. This method will fail if
|
|
||||||
/// the ID is unknown.
|
|
||||||
fn get_unique_type_id_as_string(&self, unique_type_id: UniqueTypeId) -> &str {
|
|
||||||
self.unique_id_interner.get(unique_type_id)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Gets the `UniqueTypeId` for the given type. If the `UniqueTypeId` for the given
|
|
||||||
/// type has been requested before, this is just a table lookup. Otherwise, an
|
|
||||||
/// ID will be generated and stored for later lookup.
|
|
||||||
fn get_unique_type_id_of_type<'a>(
|
|
||||||
&mut self,
|
|
||||||
cx: &CodegenCx<'a, 'tcx>,
|
|
||||||
type_: Ty<'tcx>,
|
|
||||||
) -> UniqueTypeId {
|
|
||||||
// Let's see if we already have something in the cache.
|
|
||||||
if let Some(unique_type_id) = self.type_to_unique_id.get(&type_).cloned() {
|
|
||||||
return unique_type_id;
|
|
||||||
}
|
|
||||||
// If not, generate one.
|
|
||||||
|
|
||||||
// The hasher we are using to generate the UniqueTypeId. We want
|
|
||||||
// something that provides more than the 64 bits of the DefaultHasher.
|
|
||||||
let mut hasher = StableHasher::new();
|
|
||||||
let mut hcx = cx.tcx.create_stable_hashing_context();
|
|
||||||
let type_ = cx.tcx.erase_regions(type_);
|
|
||||||
hcx.while_hashing_spans(false, |hcx| {
|
|
||||||
hcx.with_node_id_hashing_mode(NodeIdHashingMode::HashDefPath, |hcx| {
|
|
||||||
type_.hash_stable(hcx, &mut hasher);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
let unique_type_id = hasher.finish::<Fingerprint>().to_hex();
|
|
||||||
|
|
||||||
let key = self.unique_id_interner.intern(&unique_type_id);
|
|
||||||
self.type_to_unique_id.insert(type_, key);
|
|
||||||
|
|
||||||
key
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Gets the `UniqueTypeId` for an enum variant. Enum variants are not really
|
|
||||||
/// types of their own, so they need special handling. We still need a
|
|
||||||
/// `UniqueTypeId` for them, since to debuginfo they *are* real types.
|
|
||||||
fn get_unique_type_id_of_enum_variant<'a>(
|
|
||||||
&mut self,
|
|
||||||
cx: &CodegenCx<'a, 'tcx>,
|
|
||||||
enum_type: Ty<'tcx>,
|
|
||||||
variant_name: &str,
|
|
||||||
) -> UniqueTypeId {
|
|
||||||
let enum_type_id = self.get_unique_type_id_of_type(cx, enum_type);
|
|
||||||
let enum_variant_type_id =
|
|
||||||
format!("{}::{}", self.get_unique_type_id_as_string(enum_type_id), variant_name);
|
|
||||||
let interner_key = self.unique_id_interner.intern(&enum_variant_type_id);
|
|
||||||
interner_key
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Gets the unique type ID string for an enum variant part.
|
|
||||||
/// Variant parts are not types and shouldn't really have their own ID,
|
|
||||||
/// but it makes `set_members_of_composite_type()` simpler.
|
|
||||||
fn get_unique_type_id_str_of_enum_variant_part(
|
|
||||||
&mut self,
|
|
||||||
enum_type_id: UniqueTypeId,
|
|
||||||
) -> String {
|
|
||||||
format!("{}_variant_part", self.get_unique_type_id_as_string(enum_type_id))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Gets the `UniqueTypeId` for the type of a vtable.
|
|
||||||
fn get_unique_type_id_of_vtable_type(&mut self, vtable_type_name: &str) -> UniqueTypeId {
|
|
||||||
let interner_key = self.unique_id_interner.intern(vtable_type_name);
|
|
||||||
interner_key
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -293,7 +213,7 @@ impl<'ll, 'tcx> TypeMap<'ll, 'tcx> {
|
||||||
enum RecursiveTypeDescription<'ll, 'tcx> {
|
enum RecursiveTypeDescription<'ll, 'tcx> {
|
||||||
UnfinishedMetadata {
|
UnfinishedMetadata {
|
||||||
unfinished_type: Ty<'tcx>,
|
unfinished_type: Ty<'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
metadata_stub: &'ll DICompositeType,
|
metadata_stub: &'ll DICompositeType,
|
||||||
member_holding_stub: &'ll DICompositeType,
|
member_holding_stub: &'ll DICompositeType,
|
||||||
member_description_factory: MemberDescriptionFactory<'ll, 'tcx>,
|
member_description_factory: MemberDescriptionFactory<'ll, 'tcx>,
|
||||||
|
@ -304,15 +224,13 @@ enum RecursiveTypeDescription<'ll, 'tcx> {
|
||||||
fn create_and_register_recursive_type_forward_declaration<'ll, 'tcx>(
|
fn create_and_register_recursive_type_forward_declaration<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
unfinished_type: Ty<'tcx>,
|
unfinished_type: Ty<'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
metadata_stub: &'ll DICompositeType,
|
metadata_stub: &'ll DICompositeType,
|
||||||
member_holding_stub: &'ll DICompositeType,
|
member_holding_stub: &'ll DICompositeType,
|
||||||
member_description_factory: MemberDescriptionFactory<'ll, 'tcx>,
|
member_description_factory: MemberDescriptionFactory<'ll, 'tcx>,
|
||||||
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
||||||
// Insert the stub into the `TypeMap` in order to allow for recursive references.
|
// Insert the stub into the `TypeMap` in order to allow for recursive references.
|
||||||
let mut type_map = debug_context(cx).type_map.borrow_mut();
|
debug_context(cx).type_map.register_unique_id_with_metadata(unique_type_id, metadata_stub);
|
||||||
type_map.register_unique_id_with_metadata(unique_type_id, metadata_stub);
|
|
||||||
type_map.register_type_with_metadata(unfinished_type, metadata_stub);
|
|
||||||
|
|
||||||
UnfinishedMetadata {
|
UnfinishedMetadata {
|
||||||
unfinished_type,
|
unfinished_type,
|
||||||
|
@ -344,9 +262,10 @@ impl<'ll, 'tcx> RecursiveTypeDescription<'ll, 'tcx> {
|
||||||
// `create_and_register_recursive_type_forward_declaration()`
|
// `create_and_register_recursive_type_forward_declaration()`
|
||||||
// function.
|
// function.
|
||||||
{
|
{
|
||||||
let type_map = debug_context(cx).type_map.borrow();
|
if debug_context(cx)
|
||||||
if type_map.find_metadata_for_unique_id(unique_type_id).is_none()
|
.type_map
|
||||||
|| type_map.find_metadata_for_type(unfinished_type).is_none()
|
.find_metadata_for_unique_id(unique_type_id)
|
||||||
|
.is_none()
|
||||||
{
|
{
|
||||||
bug!(
|
bug!(
|
||||||
"Forward declaration of potentially recursive type \
|
"Forward declaration of potentially recursive type \
|
||||||
|
@ -379,7 +298,7 @@ impl<'ll, 'tcx> RecursiveTypeDescription<'ll, 'tcx> {
|
||||||
macro_rules! return_if_metadata_created_in_meantime {
|
macro_rules! return_if_metadata_created_in_meantime {
|
||||||
($cx: expr, $unique_type_id: expr) => {
|
($cx: expr, $unique_type_id: expr) => {
|
||||||
if let Some(metadata) =
|
if let Some(metadata) =
|
||||||
debug_context($cx).type_map.borrow().find_metadata_for_unique_id($unique_type_id)
|
debug_context($cx).type_map.find_metadata_for_unique_id($unique_type_id)
|
||||||
{
|
{
|
||||||
return MetadataCreationResult::new(metadata, true);
|
return MetadataCreationResult::new(metadata, true);
|
||||||
}
|
}
|
||||||
|
@ -390,7 +309,7 @@ macro_rules! return_if_metadata_created_in_meantime {
|
||||||
/// For slices (that is, "arrays" of unknown size) use [slice_type_metadata].
|
/// For slices (that is, "arrays" of unknown size) use [slice_type_metadata].
|
||||||
fn fixed_size_array_metadata<'ll, 'tcx>(
|
fn fixed_size_array_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
array_type: Ty<'tcx>,
|
array_type: Ty<'tcx>,
|
||||||
) -> MetadataCreationResult<'ll> {
|
) -> MetadataCreationResult<'ll> {
|
||||||
let ty::Array(element_type, len) = array_type.kind() else {
|
let ty::Array(element_type, len) = array_type.kind() else {
|
||||||
|
@ -434,7 +353,7 @@ fn pointer_or_reference_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
ptr_type: Ty<'tcx>,
|
ptr_type: Ty<'tcx>,
|
||||||
pointee_type: Ty<'tcx>,
|
pointee_type: Ty<'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
) -> MetadataCreationResult<'ll> {
|
) -> MetadataCreationResult<'ll> {
|
||||||
let pointee_type_metadata = type_metadata(cx, pointee_type);
|
let pointee_type_metadata = type_metadata(cx, pointee_type);
|
||||||
|
|
||||||
|
@ -531,16 +450,42 @@ fn pointer_or_reference_metadata<'ll, 'tcx>(
|
||||||
|
|
||||||
fn subroutine_type_metadata<'ll, 'tcx>(
|
fn subroutine_type_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
signature: ty::PolyFnSig<'tcx>,
|
|
||||||
) -> MetadataCreationResult<'ll> {
|
) -> MetadataCreationResult<'ll> {
|
||||||
let signature =
|
// It's possible to create a self-referential
|
||||||
cx.tcx.normalize_erasing_late_bound_regions(ty::ParamEnv::reveal_all(), signature);
|
// type in Rust by using 'impl trait':
|
||||||
|
//
|
||||||
|
// fn foo() -> impl Copy { foo }
|
||||||
|
//
|
||||||
|
// Unfortunately LLVM's API does not allow us to create recursive subroutine types.
|
||||||
|
// In order to work around that restriction we place a marker type in the type map,
|
||||||
|
// before creating the actual type. If the actual type is recursive, it will hit the
|
||||||
|
// marker type. So we end up with a type that looks like
|
||||||
|
//
|
||||||
|
// fn foo() -> <recursive_type>
|
||||||
|
//
|
||||||
|
// Once that is created, we replace the marker in the typemap with the actual type.
|
||||||
|
debug_context(cx)
|
||||||
|
.type_map
|
||||||
|
.unique_id_to_metadata
|
||||||
|
.borrow_mut()
|
||||||
|
.insert(unique_type_id, recursion_marker_type(cx));
|
||||||
|
|
||||||
let signature_metadata: Vec<_> = iter::once(
|
let UniqueTypeId::Ty(fn_ty, _) = unique_type_id else {
|
||||||
|
bug!("subroutine_type_metadata() called with unexpected input type: {:?}", unique_type_id)
|
||||||
|
};
|
||||||
|
|
||||||
|
let signature = cx
|
||||||
|
.tcx
|
||||||
|
.normalize_erasing_late_bound_regions(ty::ParamEnv::reveal_all(), fn_ty.fn_sig(cx.tcx));
|
||||||
|
|
||||||
|
let signature_metadata: SmallVec<[_; 32]> = iter::once(
|
||||||
// return type
|
// return type
|
||||||
match signature.output().kind() {
|
match signature.output().kind() {
|
||||||
ty::Tuple(tys) if tys.is_empty() => None,
|
ty::Tuple(tys) if tys.is_empty() => {
|
||||||
|
// this is a "void" function
|
||||||
|
None
|
||||||
|
}
|
||||||
_ => Some(type_metadata(cx, signature.output())),
|
_ => Some(type_metadata(cx, signature.output())),
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
@ -550,17 +495,30 @@ fn subroutine_type_metadata<'ll, 'tcx>(
|
||||||
)
|
)
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
return_if_metadata_created_in_meantime!(cx, unique_type_id);
|
debug_context(cx).type_map.unique_id_to_metadata.borrow_mut().remove(&unique_type_id);
|
||||||
|
|
||||||
MetadataCreationResult::new(
|
let fn_metadata = unsafe {
|
||||||
unsafe {
|
llvm::LLVMRustDIBuilderCreateSubroutineType(
|
||||||
llvm::LLVMRustDIBuilderCreateSubroutineType(
|
DIB(cx),
|
||||||
DIB(cx),
|
create_DIArray(DIB(cx), &signature_metadata[..]),
|
||||||
create_DIArray(DIB(cx), &signature_metadata[..]),
|
)
|
||||||
)
|
};
|
||||||
},
|
|
||||||
false,
|
// This is actually a function pointer, so wrap it in pointer DI.
|
||||||
)
|
let name = compute_debuginfo_type_name(cx.tcx, fn_ty, false);
|
||||||
|
let metadata = unsafe {
|
||||||
|
llvm::LLVMRustDIBuilderCreatePointerType(
|
||||||
|
DIB(cx),
|
||||||
|
fn_metadata,
|
||||||
|
cx.tcx.data_layout.pointer_size.bits(),
|
||||||
|
cx.tcx.data_layout.pointer_align.abi.bits() as u32,
|
||||||
|
0, // Ignore DWARF address space.
|
||||||
|
name.as_ptr().cast(),
|
||||||
|
name.len(),
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
MetadataCreationResult::new(metadata, false)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create debuginfo for `dyn SomeTrait` types. Currently these are empty structs
|
/// Create debuginfo for `dyn SomeTrait` types. Currently these are empty structs
|
||||||
|
@ -568,7 +526,7 @@ fn subroutine_type_metadata<'ll, 'tcx>(
|
||||||
fn dyn_type_metadata<'ll, 'tcx>(
|
fn dyn_type_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
dyn_type: Ty<'tcx>,
|
dyn_type: Ty<'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
) -> &'ll DIType {
|
) -> &'ll DIType {
|
||||||
if let ty::Dynamic(..) = dyn_type.kind() {
|
if let ty::Dynamic(..) = dyn_type.kind() {
|
||||||
let type_name = compute_debuginfo_type_name(cx.tcx, dyn_type, true);
|
let type_name = compute_debuginfo_type_name(cx.tcx, dyn_type, true);
|
||||||
|
@ -598,7 +556,7 @@ fn dyn_type_metadata<'ll, 'tcx>(
|
||||||
fn slice_type_metadata<'ll, 'tcx>(
|
fn slice_type_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
slice_type: Ty<'tcx>,
|
slice_type: Ty<'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
) -> MetadataCreationResult<'ll> {
|
) -> MetadataCreationResult<'ll> {
|
||||||
let element_type = match slice_type.kind() {
|
let element_type = match slice_type.kind() {
|
||||||
ty::Slice(element_type) => *element_type,
|
ty::Slice(element_type) => *element_type,
|
||||||
|
@ -617,38 +575,11 @@ fn slice_type_metadata<'ll, 'tcx>(
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn type_metadata<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>, t: Ty<'tcx>) -> &'ll DIType {
|
pub fn type_metadata<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>, t: Ty<'tcx>) -> &'ll DIType {
|
||||||
// Get the unique type ID of this type.
|
let unique_type_id = UniqueTypeId::for_ty(cx.tcx, t);
|
||||||
let unique_type_id = {
|
|
||||||
let mut type_map = debug_context(cx).type_map.borrow_mut();
|
if let Some(metadata) = debug_context(cx).type_map.find_metadata_for_unique_id(unique_type_id) {
|
||||||
// First, try to find the type in `TypeMap`. If we have seen it before, we
|
return metadata;
|
||||||
// can exit early here.
|
}
|
||||||
match type_map.find_metadata_for_type(t) {
|
|
||||||
Some(metadata) => {
|
|
||||||
return metadata;
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
// The Ty is not in the `TypeMap` but maybe we have already seen
|
|
||||||
// an equivalent type (e.g., only differing in region arguments).
|
|
||||||
// In order to find out, generate the unique type ID and look
|
|
||||||
// that up.
|
|
||||||
let unique_type_id = type_map.get_unique_type_id_of_type(cx, t);
|
|
||||||
match type_map.find_metadata_for_unique_id(unique_type_id) {
|
|
||||||
Some(metadata) => {
|
|
||||||
// There is already an equivalent type in the TypeMap.
|
|
||||||
// Register this Ty as an alias in the cache and
|
|
||||||
// return the cached metadata.
|
|
||||||
type_map.register_type_with_metadata(t, metadata);
|
|
||||||
return metadata;
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
// There really is no type metadata for this type, so
|
|
||||||
// proceed by creating it.
|
|
||||||
unique_type_id
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
debug!("type_metadata: {:?}", t);
|
debug!("type_metadata: {:?}", t);
|
||||||
|
|
||||||
|
@ -673,64 +604,7 @@ pub fn type_metadata<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>, t: Ty<'tcx>) -> &'ll
|
||||||
ty::Adt(def, _) if def.is_box() => {
|
ty::Adt(def, _) if def.is_box() => {
|
||||||
pointer_or_reference_metadata(cx, t, t.boxed_ty(), unique_type_id)
|
pointer_or_reference_metadata(cx, t, t.boxed_ty(), unique_type_id)
|
||||||
}
|
}
|
||||||
ty::FnDef(..) | ty::FnPtr(_) => {
|
ty::FnDef(..) | ty::FnPtr(_) => subroutine_type_metadata(cx, unique_type_id),
|
||||||
if let Some(metadata) =
|
|
||||||
debug_context(cx).type_map.borrow().find_metadata_for_unique_id(unique_type_id)
|
|
||||||
{
|
|
||||||
return metadata;
|
|
||||||
}
|
|
||||||
|
|
||||||
// It's possible to create a self-referential
|
|
||||||
// type in Rust by using 'impl trait':
|
|
||||||
//
|
|
||||||
// fn foo() -> impl Copy { foo }
|
|
||||||
//
|
|
||||||
// See `TypeMap::remove_type` for more detals
|
|
||||||
// about the workaround.
|
|
||||||
|
|
||||||
let temp_type = {
|
|
||||||
unsafe {
|
|
||||||
// The choice of type here is pretty arbitrary -
|
|
||||||
// anything reading the debuginfo for a recursive
|
|
||||||
// type is going to see *something* weird - the only
|
|
||||||
// question is what exactly it will see.
|
|
||||||
let name = "<recur_type>";
|
|
||||||
llvm::LLVMRustDIBuilderCreateBasicType(
|
|
||||||
DIB(cx),
|
|
||||||
name.as_ptr().cast(),
|
|
||||||
name.len(),
|
|
||||||
cx.size_of(t).bits(),
|
|
||||||
DW_ATE_unsigned,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let type_map = &debug_context(cx).type_map;
|
|
||||||
type_map.borrow_mut().register_type_with_metadata(t, temp_type);
|
|
||||||
|
|
||||||
let fn_metadata =
|
|
||||||
subroutine_type_metadata(cx, unique_type_id, t.fn_sig(cx.tcx)).metadata;
|
|
||||||
|
|
||||||
type_map.borrow_mut().remove_type(t);
|
|
||||||
|
|
||||||
// This is actually a function pointer, so wrap it in pointer DI.
|
|
||||||
let (pointer_size, pointer_align) =
|
|
||||||
cx.size_and_align_of(cx.tcx.mk_imm_ptr(cx.tcx.mk_unit()));
|
|
||||||
let name = compute_debuginfo_type_name(cx.tcx, t, false);
|
|
||||||
let md = unsafe {
|
|
||||||
llvm::LLVMRustDIBuilderCreatePointerType(
|
|
||||||
DIB(cx),
|
|
||||||
fn_metadata,
|
|
||||||
pointer_size.bits(),
|
|
||||||
pointer_align.bits() as u32,
|
|
||||||
0, // Ignore DWARF address space.
|
|
||||||
name.as_ptr().cast(),
|
|
||||||
name.len(),
|
|
||||||
)
|
|
||||||
};
|
|
||||||
|
|
||||||
MetadataCreationResult::new(md, false)
|
|
||||||
}
|
|
||||||
ty::Closure(def_id, substs) => {
|
ty::Closure(def_id, substs) => {
|
||||||
let upvar_tys: Vec<_> = substs.as_closure().upvar_tys().collect();
|
let upvar_tys: Vec<_> = substs.as_closure().upvar_tys().collect();
|
||||||
let containing_scope = get_namespace_for_item(cx, def_id);
|
let containing_scope = get_namespace_for_item(cx, def_id);
|
||||||
|
@ -762,47 +636,54 @@ pub fn type_metadata<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>, t: Ty<'tcx>) -> &'ll
|
||||||
};
|
};
|
||||||
|
|
||||||
{
|
{
|
||||||
let mut type_map = debug_context(cx).type_map.borrow_mut();
|
|
||||||
|
|
||||||
if already_stored_in_typemap {
|
if already_stored_in_typemap {
|
||||||
// Also make sure that we already have a `TypeMap` entry for the unique type ID.
|
// Make sure that we really do have a `TypeMap` entry for the unique type ID.
|
||||||
let Some(metadata_for_uid) = type_map.find_metadata_for_unique_id(unique_type_id) else {
|
let metadata_for_uid =
|
||||||
bug!(
|
match debug_context(cx).type_map.find_metadata_for_unique_id(unique_type_id) {
|
||||||
"expected type metadata for unique \
|
Some(metadata) => metadata,
|
||||||
type ID '{}' to already be in \
|
None => {
|
||||||
the `debuginfo::TypeMap` but it \
|
|
||||||
was not. (Ty = {})",
|
|
||||||
type_map.get_unique_type_id_as_string(unique_type_id),
|
|
||||||
t
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
match type_map.find_metadata_for_type(t) {
|
|
||||||
Some(metadata) => {
|
|
||||||
if metadata != metadata_for_uid {
|
|
||||||
bug!(
|
bug!(
|
||||||
"mismatch between `Ty` and \
|
"expected type metadata for unique \
|
||||||
`UniqueTypeId` maps in \
|
type ID '{:?}' to already be in \
|
||||||
`debuginfo::TypeMap`. \
|
the `debuginfo::TypeMap` but it \
|
||||||
UniqueTypeId={}, Ty={}",
|
was not. (Ty = {})",
|
||||||
type_map.get_unique_type_id_as_string(unique_type_id),
|
unique_type_id,
|
||||||
t
|
t
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
None => {
|
|
||||||
type_map.register_type_with_metadata(t, metadata);
|
debug_assert_eq!(metadata_for_uid as *const _, metadata as *const _);
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
type_map.register_type_with_metadata(t, metadata);
|
debug_context(cx).type_map.register_unique_id_with_metadata(unique_type_id, metadata);
|
||||||
type_map.register_unique_id_with_metadata(unique_type_id, metadata);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
metadata
|
metadata
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn recursion_marker_type<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>) -> &'ll DIType {
|
||||||
|
*debug_context(cx).recursion_marker_type.get_or_init(move || {
|
||||||
|
unsafe {
|
||||||
|
// The choice of type here is pretty arbitrary -
|
||||||
|
// anything reading the debuginfo for a recursive
|
||||||
|
// type is going to see *something* weird - the only
|
||||||
|
// question is what exactly it will see.
|
||||||
|
//
|
||||||
|
// FIXME: the name `<recur_type>` does not fit the naming scheme
|
||||||
|
// of other types.
|
||||||
|
let name = "<recur_type>";
|
||||||
|
llvm::LLVMRustDIBuilderCreateBasicType(
|
||||||
|
DIB(cx),
|
||||||
|
name.as_ptr().cast(),
|
||||||
|
name.len(),
|
||||||
|
cx.tcx.data_layout.pointer_size.bits(),
|
||||||
|
DW_ATE_unsigned,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
fn hex_encode(data: &[u8]) -> String {
|
fn hex_encode(data: &[u8]) -> String {
|
||||||
let mut hex_string = String::with_capacity(data.len() * 2);
|
let mut hex_string = String::with_capacity(data.len() * 2);
|
||||||
for byte in data.iter() {
|
for byte in data.iter() {
|
||||||
|
@ -983,7 +864,7 @@ fn basic_type_metadata<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>, t: Ty<'tcx>) -> &'l
|
||||||
fn foreign_type_metadata<'ll, 'tcx>(
|
fn foreign_type_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
t: Ty<'tcx>,
|
t: Ty<'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
) -> &'ll DIType {
|
) -> &'ll DIType {
|
||||||
debug!("foreign_type_metadata: {:?}", t);
|
debug!("foreign_type_metadata: {:?}", t);
|
||||||
|
|
||||||
|
@ -1304,7 +1185,7 @@ impl<'tcx> StructMemberDescriptionFactory<'tcx> {
|
||||||
fn prepare_struct_metadata<'ll, 'tcx>(
|
fn prepare_struct_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
struct_type: Ty<'tcx>,
|
struct_type: Ty<'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
||||||
let struct_name = compute_debuginfo_type_name(cx.tcx, struct_type, false);
|
let struct_name = compute_debuginfo_type_name(cx.tcx, struct_type, false);
|
||||||
|
|
||||||
|
@ -1413,7 +1294,7 @@ fn prepare_tuple_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
tuple_type: Ty<'tcx>,
|
tuple_type: Ty<'tcx>,
|
||||||
component_types: &[Ty<'tcx>],
|
component_types: &[Ty<'tcx>],
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
containing_scope: Option<&'ll DIScope>,
|
containing_scope: Option<&'ll DIScope>,
|
||||||
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
||||||
let (size, align) = cx.size_and_align_of(tuple_type);
|
let (size, align) = cx.size_and_align_of(tuple_type);
|
||||||
|
@ -1481,7 +1362,7 @@ impl<'tcx> UnionMemberDescriptionFactory<'tcx> {
|
||||||
fn prepare_union_metadata<'ll, 'tcx>(
|
fn prepare_union_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
union_type: Ty<'tcx>,
|
union_type: Ty<'tcx>,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
||||||
let union_name = compute_debuginfo_type_name(cx.tcx, union_type, false);
|
let union_name = compute_debuginfo_type_name(cx.tcx, union_type, false);
|
||||||
|
|
||||||
|
@ -1568,7 +1449,7 @@ impl<'ll, 'tcx> EnumMemberDescriptionFactory<'ll, 'tcx> {
|
||||||
};
|
};
|
||||||
|
|
||||||
let variant_info_for = |index: VariantIdx| match *self.enum_type.kind() {
|
let variant_info_for = |index: VariantIdx| match *self.enum_type.kind() {
|
||||||
ty::Adt(adt, _) => VariantInfo::Adt(&adt.variants[index]),
|
ty::Adt(adt, _) => VariantInfo::Adt(&adt.variants[index], index),
|
||||||
ty::Generator(def_id, _, _) => {
|
ty::Generator(def_id, _, _) => {
|
||||||
let (generator_layout, generator_saved_local_names) =
|
let (generator_layout, generator_saved_local_names) =
|
||||||
generator_variant_info_data.as_ref().unwrap();
|
generator_variant_info_data.as_ref().unwrap();
|
||||||
|
@ -1911,7 +1792,7 @@ impl<'tcx> VariantMemberDescriptionFactory<'tcx> {
|
||||||
|
|
||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone)]
|
||||||
enum VariantInfo<'a, 'tcx> {
|
enum VariantInfo<'a, 'tcx> {
|
||||||
Adt(&'tcx ty::VariantDef),
|
Adt(&'tcx ty::VariantDef, VariantIdx),
|
||||||
Generator {
|
Generator {
|
||||||
def_id: DefId,
|
def_id: DefId,
|
||||||
generator_layout: &'tcx GeneratorLayout<'tcx>,
|
generator_layout: &'tcx GeneratorLayout<'tcx>,
|
||||||
|
@ -1921,9 +1802,17 @@ enum VariantInfo<'a, 'tcx> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'tcx> VariantInfo<'_, 'tcx> {
|
impl<'tcx> VariantInfo<'_, 'tcx> {
|
||||||
|
fn variant_idx(&self) -> VariantIdx {
|
||||||
|
match self {
|
||||||
|
VariantInfo::Adt(_, variant_index) | VariantInfo::Generator { variant_index, .. } => {
|
||||||
|
*variant_index
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn map_struct_name<R>(&self, f: impl FnOnce(&str) -> R) -> R {
|
fn map_struct_name<R>(&self, f: impl FnOnce(&str) -> R) -> R {
|
||||||
match self {
|
match self {
|
||||||
VariantInfo::Adt(variant) => f(variant.name.as_str()),
|
VariantInfo::Adt(variant, _) => f(variant.name.as_str()),
|
||||||
VariantInfo::Generator { variant_index, .. } => {
|
VariantInfo::Generator { variant_index, .. } => {
|
||||||
f(&GeneratorSubsts::variant_name(*variant_index))
|
f(&GeneratorSubsts::variant_name(*variant_index))
|
||||||
}
|
}
|
||||||
|
@ -1932,7 +1821,7 @@ impl<'tcx> VariantInfo<'_, 'tcx> {
|
||||||
|
|
||||||
fn variant_name(&self) -> String {
|
fn variant_name(&self) -> String {
|
||||||
match self {
|
match self {
|
||||||
VariantInfo::Adt(variant) => variant.name.to_string(),
|
VariantInfo::Adt(variant, _) => variant.name.to_string(),
|
||||||
VariantInfo::Generator { variant_index, .. } => {
|
VariantInfo::Generator { variant_index, .. } => {
|
||||||
// Since GDB currently prints out the raw discriminant along
|
// Since GDB currently prints out the raw discriminant along
|
||||||
// with every variant, make each variant name be just the value
|
// with every variant, make each variant name be just the value
|
||||||
|
@ -1945,7 +1834,7 @@ impl<'tcx> VariantInfo<'_, 'tcx> {
|
||||||
|
|
||||||
fn field_name(&self, i: usize) -> String {
|
fn field_name(&self, i: usize) -> String {
|
||||||
let field_name = match *self {
|
let field_name = match *self {
|
||||||
VariantInfo::Adt(variant) if variant.ctor_kind != CtorKind::Fn => {
|
VariantInfo::Adt(variant, _) if variant.ctor_kind != CtorKind::Fn => {
|
||||||
Some(variant.fields[i].name)
|
Some(variant.fields[i].name)
|
||||||
}
|
}
|
||||||
VariantInfo::Generator {
|
VariantInfo::Generator {
|
||||||
|
@ -1986,10 +1875,8 @@ fn describe_enum_variant<'ll, 'tcx>(
|
||||||
containing_scope: &'ll DIScope,
|
containing_scope: &'ll DIScope,
|
||||||
) -> (&'ll DICompositeType, MemberDescriptionFactory<'ll, 'tcx>) {
|
) -> (&'ll DICompositeType, MemberDescriptionFactory<'ll, 'tcx>) {
|
||||||
let metadata_stub = variant.map_struct_name(|variant_name| {
|
let metadata_stub = variant.map_struct_name(|variant_name| {
|
||||||
let unique_type_id = debug_context(cx)
|
let unique_type_id =
|
||||||
.type_map
|
UniqueTypeId::for_enum_variant(cx.tcx, layout.ty, variant.variant_idx());
|
||||||
.borrow_mut()
|
|
||||||
.get_unique_type_id_of_enum_variant(cx, layout.ty, variant_name);
|
|
||||||
|
|
||||||
let (size, align) = cx.size_and_align_of(layout.ty);
|
let (size, align) = cx.size_and_align_of(layout.ty);
|
||||||
|
|
||||||
|
@ -2019,7 +1906,7 @@ fn prepare_enum_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
enum_type: Ty<'tcx>,
|
enum_type: Ty<'tcx>,
|
||||||
enum_def_id: DefId,
|
enum_def_id: DefId,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
outer_field_tys: Vec<Ty<'tcx>>,
|
outer_field_tys: Vec<Ty<'tcx>>,
|
||||||
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
) -> RecursiveTypeDescription<'ll, 'tcx> {
|
||||||
let tcx = cx.tcx;
|
let tcx = cx.tcx;
|
||||||
|
@ -2143,8 +2030,7 @@ fn prepare_enum_metadata<'ll, 'tcx>(
|
||||||
};
|
};
|
||||||
|
|
||||||
let enum_metadata = {
|
let enum_metadata = {
|
||||||
let type_map = debug_context(cx).type_map.borrow();
|
let unique_type_id_str = unique_type_id.to_string(tcx);
|
||||||
let unique_type_id_str = type_map.get_unique_type_id_as_string(unique_type_id);
|
|
||||||
|
|
||||||
unsafe {
|
unsafe {
|
||||||
llvm::LLVMRustDIBuilderCreateUnionType(
|
llvm::LLVMRustDIBuilderCreateUnionType(
|
||||||
|
@ -2255,10 +2141,9 @@ fn prepare_enum_metadata<'ll, 'tcx>(
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let variant_part_unique_type_id_str = debug_context(cx)
|
let variant_part_unique_type_id_str =
|
||||||
.type_map
|
UniqueTypeId::for_enum_variant_part(tcx, enum_type).to_string(tcx);
|
||||||
.borrow_mut()
|
|
||||||
.get_unique_type_id_str_of_enum_variant_part(unique_type_id);
|
|
||||||
let empty_array = create_DIArray(DIB(cx), &[]);
|
let empty_array = create_DIArray(DIB(cx), &[]);
|
||||||
let name = "";
|
let name = "";
|
||||||
let variant_part = unsafe {
|
let variant_part = unsafe {
|
||||||
|
@ -2286,9 +2171,7 @@ fn prepare_enum_metadata<'ll, 'tcx>(
|
||||||
// an equivalent layout but offers us much better integration with
|
// an equivalent layout but offers us much better integration with
|
||||||
// debuggers.
|
// debuggers.
|
||||||
let type_array = create_DIArray(DIB(cx), &[Some(variant_part)]);
|
let type_array = create_DIArray(DIB(cx), &[Some(variant_part)]);
|
||||||
|
let unique_type_id_str = unique_type_id.to_string(tcx);
|
||||||
let type_map = debug_context(cx).type_map.borrow();
|
|
||||||
let unique_type_id_str = type_map.get_unique_type_id_as_string(unique_type_id);
|
|
||||||
|
|
||||||
unsafe {
|
unsafe {
|
||||||
llvm::LLVMRustDIBuilderCreateStructType(
|
llvm::LLVMRustDIBuilderCreateStructType(
|
||||||
|
@ -2334,7 +2217,7 @@ fn composite_type_metadata<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
composite_type: Ty<'tcx>,
|
composite_type: Ty<'tcx>,
|
||||||
composite_type_name: &str,
|
composite_type_name: &str,
|
||||||
composite_type_unique_id: UniqueTypeId,
|
composite_type_unique_id: UniqueTypeId<'tcx>,
|
||||||
member_descriptions: Vec<MemberDescription<'ll>>,
|
member_descriptions: Vec<MemberDescription<'ll>>,
|
||||||
containing_scope: Option<&'ll DIScope>,
|
containing_scope: Option<&'ll DIScope>,
|
||||||
) -> &'ll DICompositeType {
|
) -> &'ll DICompositeType {
|
||||||
|
@ -2457,13 +2340,12 @@ fn create_struct_stub<'ll, 'tcx>(
|
||||||
size: Size,
|
size: Size,
|
||||||
align: Align,
|
align: Align,
|
||||||
type_name: &str,
|
type_name: &str,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
containing_scope: Option<&'ll DIScope>,
|
containing_scope: Option<&'ll DIScope>,
|
||||||
flags: DIFlags,
|
flags: DIFlags,
|
||||||
vtable_holder: Option<&'ll DIType>,
|
vtable_holder: Option<&'ll DIType>,
|
||||||
) -> &'ll DICompositeType {
|
) -> &'ll DICompositeType {
|
||||||
let type_map = debug_context(cx).type_map.borrow();
|
let unique_type_id = unique_type_id.to_string(cx.tcx);
|
||||||
let unique_type_id = type_map.get_unique_type_id_as_string(unique_type_id);
|
|
||||||
|
|
||||||
let metadata_stub = unsafe {
|
let metadata_stub = unsafe {
|
||||||
// `LLVMRustDIBuilderCreateStructType()` wants an empty array. A null
|
// `LLVMRustDIBuilderCreateStructType()` wants an empty array. A null
|
||||||
|
@ -2497,13 +2379,11 @@ fn create_union_stub<'ll, 'tcx>(
|
||||||
cx: &CodegenCx<'ll, 'tcx>,
|
cx: &CodegenCx<'ll, 'tcx>,
|
||||||
union_type: Ty<'tcx>,
|
union_type: Ty<'tcx>,
|
||||||
union_type_name: &str,
|
union_type_name: &str,
|
||||||
unique_type_id: UniqueTypeId,
|
unique_type_id: UniqueTypeId<'tcx>,
|
||||||
containing_scope: &'ll DIScope,
|
containing_scope: &'ll DIScope,
|
||||||
) -> &'ll DICompositeType {
|
) -> &'ll DICompositeType {
|
||||||
let (union_size, union_align) = cx.size_and_align_of(union_type);
|
let (union_size, union_align) = cx.size_and_align_of(union_type);
|
||||||
|
let unique_type_id = unique_type_id.to_string(cx.tcx);
|
||||||
let type_map = debug_context(cx).type_map.borrow();
|
|
||||||
let unique_type_id = type_map.get_unique_type_id_as_string(unique_type_id);
|
|
||||||
|
|
||||||
let metadata_stub = unsafe {
|
let metadata_stub = unsafe {
|
||||||
// `LLVMRustDIBuilderCreateUnionType()` wants an empty array. A null
|
// `LLVMRustDIBuilderCreateUnionType()` wants an empty array. A null
|
||||||
|
@ -2627,10 +2507,7 @@ fn vtable_type_metadata<'ll, 'tcx>(
|
||||||
|
|
||||||
let vtable_type_name =
|
let vtable_type_name =
|
||||||
compute_debuginfo_vtable_name(cx.tcx, ty, poly_trait_ref, VTableNameKind::Type);
|
compute_debuginfo_vtable_name(cx.tcx, ty, poly_trait_ref, VTableNameKind::Type);
|
||||||
let unique_type_id = debug_context(cx)
|
let unique_type_id = UniqueTypeId::for_vtable_ty(tcx, ty, poly_trait_ref);
|
||||||
.type_map
|
|
||||||
.borrow_mut()
|
|
||||||
.get_unique_type_id_of_vtable_type(&vtable_type_name);
|
|
||||||
let size = pointer_size * vtable_entries.len() as u64;
|
let size = pointer_size * vtable_entries.len() as u64;
|
||||||
|
|
||||||
// This gets mapped to a DW_AT_containing_type attribute which allows GDB to correlate
|
// This gets mapped to a DW_AT_containing_type attribute which allows GDB to correlate
|
||||||
|
|
|
@ -38,6 +38,7 @@ use libc::c_uint;
|
||||||
use smallvec::SmallVec;
|
use smallvec::SmallVec;
|
||||||
use std::cell::RefCell;
|
use std::cell::RefCell;
|
||||||
use std::iter;
|
use std::iter;
|
||||||
|
use std::lazy::OnceCell;
|
||||||
use tracing::debug;
|
use tracing::debug;
|
||||||
|
|
||||||
mod create_scope_map;
|
mod create_scope_map;
|
||||||
|
@ -63,9 +64,11 @@ pub struct CrateDebugContext<'a, 'tcx> {
|
||||||
created_files: RefCell<FxHashMap<(Option<String>, Option<String>), &'a DIFile>>,
|
created_files: RefCell<FxHashMap<(Option<String>, Option<String>), &'a DIFile>>,
|
||||||
created_enum_disr_types: RefCell<FxHashMap<(DefId, Primitive), &'a DIType>>,
|
created_enum_disr_types: RefCell<FxHashMap<(DefId, Primitive), &'a DIType>>,
|
||||||
|
|
||||||
type_map: RefCell<TypeMap<'a, 'tcx>>,
|
type_map: TypeMap<'a, 'tcx>,
|
||||||
namespace_map: RefCell<DefIdMap<&'a DIScope>>,
|
namespace_map: RefCell<DefIdMap<&'a DIScope>>,
|
||||||
|
|
||||||
|
recursion_marker_type: OnceCell<&'a DIType>,
|
||||||
|
|
||||||
// This collection is used to assert that composite types (structs, enums,
|
// This collection is used to assert that composite types (structs, enums,
|
||||||
// ...) have their members only set once:
|
// ...) have their members only set once:
|
||||||
composite_types_completed: RefCell<FxHashSet<&'a DIType>>,
|
composite_types_completed: RefCell<FxHashSet<&'a DIType>>,
|
||||||
|
@ -93,6 +96,7 @@ impl<'a, 'tcx> CrateDebugContext<'a, 'tcx> {
|
||||||
created_enum_disr_types: Default::default(),
|
created_enum_disr_types: Default::default(),
|
||||||
type_map: Default::default(),
|
type_map: Default::default(),
|
||||||
namespace_map: RefCell::new(Default::default()),
|
namespace_map: RefCell::new(Default::default()),
|
||||||
|
recursion_marker_type: OnceCell::new(),
|
||||||
composite_types_completed: Default::default(),
|
composite_types_completed: Default::default(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,10 +9,14 @@
|
||||||
#![feature(crate_visibility_modifier)]
|
#![feature(crate_visibility_modifier)]
|
||||||
#![feature(let_else)]
|
#![feature(let_else)]
|
||||||
#![feature(extern_types)]
|
#![feature(extern_types)]
|
||||||
|
#![feature(once_cell)]
|
||||||
#![feature(nll)]
|
#![feature(nll)]
|
||||||
#![recursion_limit = "256"]
|
#![recursion_limit = "256"]
|
||||||
#![cfg_attr(not(bootstrap), allow(rustc::potential_query_instability))]
|
#![cfg_attr(not(bootstrap), allow(rustc::potential_query_instability))]
|
||||||
|
|
||||||
|
#[macro_use]
|
||||||
|
extern crate rustc_macros;
|
||||||
|
|
||||||
use back::write::{create_informational_target_machine, create_target_machine};
|
use back::write::{create_informational_target_machine, create_target_machine};
|
||||||
|
|
||||||
pub use llvm_util::target_features;
|
pub use llvm_util::target_features;
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
// NONMSVC: {{.*}}DIDerivedType(tag: DW_TAG_pointer_type, name: "fn() -> <recursive_type>",{{.*}}
|
// NONMSVC: {{.*}}DIDerivedType(tag: DW_TAG_pointer_type, name: "fn() -> <recursive_type>",{{.*}}
|
||||||
//
|
//
|
||||||
// CHECK: {{.*}}DISubroutineType{{.*}}
|
// CHECK: {{.*}}DISubroutineType{{.*}}
|
||||||
// CHECK: {{.*}}DIBasicType(name: "<recur_type>", encoding: DW_ATE_unsigned)
|
// CHECK: {{.*}}DIBasicType(name: "<recur_type>", size: {{32|64}}, encoding: DW_ATE_unsigned)
|
||||||
|
|
||||||
pub fn foo() -> impl Copy {
|
pub fn foo() -> impl Copy {
|
||||||
foo
|
foo
|
||||||
|
|
Loading…
Add table
Reference in a new issue