Skip to main content

rustc_codegen_llvm/debuginfo/
metadata.rs

1use std::borrow::Cow;
2use std::fmt::{self, Write};
3use std::hash::{Hash, Hasher};
4use std::path::PathBuf;
5use std::sync::Arc;
6use std::{iter, ptr};
7
8use libc::{c_longlong, c_uint};
9use rustc_abi::{Align, Size};
10use rustc_codegen_ssa::debuginfo::type_names::{VTableNameKind, cpp_like_debuginfo};
11use rustc_codegen_ssa::traits::*;
12use rustc_hir::def::{CtorKind, DefKind};
13use rustc_hir::def_id::{DefId, LOCAL_CRATE};
14use rustc_middle::bug;
15use rustc_middle::ty::layout::{
16    HasTypingEnv, LayoutOf, TyAndLayout, WIDE_PTR_ADDR, WIDE_PTR_EXTRA,
17};
18use rustc_middle::ty::{
19    self, AdtKind, CoroutineArgsExt, ExistentialTraitRef, Instance, Ty, TyCtxt, Visibility,
20};
21use rustc_session::config::{self, DebugInfo, Lto};
22use rustc_span::{DUMMY_SP, FileName, RemapPathScopeComponents, SourceFile, Span, Symbol, hygiene};
23use rustc_symbol_mangling::typeid_for_trait_ref;
24use rustc_target::spec::DebuginfoKind;
25use smallvec::smallvec;
26use tracing::{debug, instrument};
27
28pub(crate) use self::type_map::TypeMap;
29use self::type_map::{DINodeCreationResult, Stub, UniqueTypeId};
30use super::CodegenUnitDebugContext;
31use super::namespace::mangled_name_of_instance;
32use super::type_names::{compute_debuginfo_type_name, compute_debuginfo_vtable_name};
33use super::utils::{DIB, debug_context, get_namespace_for_item, is_node_local_to_unit};
34use crate::common::{AsCCharPtr, CodegenCx};
35use crate::debuginfo::metadata::type_map::build_type_with_children;
36use crate::debuginfo::utils::{WidePtrKind, wide_pointer_kind};
37use crate::debuginfo::{DIBuilderExt, dwarf_const};
38use crate::llvm::debuginfo::{
39    DIBasicType, DIBuilder, DICompositeType, DIDescriptor, DIFile, DIFlags, DILexicalBlock,
40    DIScope, DIType, DebugEmissionKind, DebugNameTableKind,
41};
42use crate::llvm::{self, FromGeneric, Value};
43
44impl PartialEq for llvm::Metadata {
45    fn eq(&self, other: &Self) -> bool {
46        ptr::eq(self, other)
47    }
48}
49
50impl Eq for llvm::Metadata {}
51
52impl Hash for llvm::Metadata {
53    fn hash<H: Hasher>(&self, hasher: &mut H) {
54        (self as *const Self).hash(hasher);
55    }
56}
57
58impl fmt::Debug for llvm::Metadata {
59    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
60        (self as *const Self).fmt(f)
61    }
62}
63
64pub(super) const UNKNOWN_LINE_NUMBER: c_uint = 0;
65pub(super) const UNKNOWN_COLUMN_NUMBER: c_uint = 0;
66
67const NO_SCOPE_METADATA: Option<&DIScope> = None;
68/// A function that returns an empty list of generic parameter debuginfo nodes.
69const NO_GENERICS: for<'ll> fn(&CodegenCx<'ll, '_>) -> SmallVec<Option<&'ll DIType>> =
70    |_| SmallVec::new();
71
72// SmallVec is used quite a bit in this module, so create a shorthand.
73// The actual number of elements is not so important.
74type SmallVec<T> = smallvec::SmallVec<[T; 16]>;
75
76mod enums;
77mod type_map;
78
79/// Returns from the enclosing function if the type debuginfo node with the given
80/// unique ID can be found in the type map.
81macro_rules! return_if_di_node_created_in_meantime {
82    ($cx: expr, $unique_type_id: expr) => {
83        if let Some(di_node) = debug_context($cx).type_map.di_node_for_unique_id($unique_type_id) {
84            return DINodeCreationResult::new(di_node, true);
85        }
86    };
87}
88
89/// Extract size and alignment from a TyAndLayout.
90#[inline]
91fn size_and_align_of(ty_and_layout: TyAndLayout<'_>) -> (Size, Align) {
92    (ty_and_layout.size, ty_and_layout.align.abi)
93}
94
95/// Creates debuginfo for a fixed size array (e.g. `[u64; 123]`).
96/// For slices (that is, "arrays" of unknown size) use [build_slice_type_di_node].
97fn build_fixed_size_array_di_node<'ll, 'tcx>(
98    cx: &CodegenCx<'ll, 'tcx>,
99    unique_type_id: UniqueTypeId<'tcx>,
100    array_type: Ty<'tcx>,
101    span: Span,
102) -> DINodeCreationResult<'ll> {
103    let ty::Array(element_type, len) = array_type.kind() else {
104        ::rustc_middle::util::bug::bug_fmt(format_args!("build_fixed_size_array_di_node() called with non-ty::Array type `{0:?}`",
        array_type))bug!("build_fixed_size_array_di_node() called with non-ty::Array type `{:?}`", array_type)
105    };
106
107    let element_type_di_node = spanned_type_di_node(cx, *element_type, span);
108
109    if let Some(di_node) =
        debug_context(cx).type_map.di_node_for_unique_id(unique_type_id) {
    return DINodeCreationResult::new(di_node, true);
};return_if_di_node_created_in_meantime!(cx, unique_type_id);
110
111    let (size, align) = cx.spanned_size_and_align_of(array_type, span);
112
113    let upper_bound = len
114        .try_to_target_usize(cx.tcx)
115        .expect("expected monomorphic const in codegen") as c_longlong;
116
117    let subrange = unsafe { llvm::LLVMDIBuilderGetOrCreateSubrange(DIB(cx), 0, upper_bound) };
118    let subscripts = &[subrange];
119
120    let di_node = unsafe {
121        llvm::LLVMDIBuilderCreateArrayType(
122            DIB(cx),
123            size.bits(),
124            align.bits() as u32,
125            element_type_di_node,
126            subscripts.as_ptr(),
127            subscripts.len() as c_uint,
128        )
129    };
130
131    DINodeCreationResult::new(di_node, false)
132}
133
134/// Creates debuginfo for built-in pointer-like things:
135///
136///  - ty::Ref
137///  - ty::RawPtr
138///  - ty::Adt in the case it's Box
139///
140/// At some point we might want to remove the special handling of Box
141/// and treat it the same as other smart pointers (like Rc, Arc, ...).
142fn build_pointer_or_reference_di_node<'ll, 'tcx>(
143    cx: &CodegenCx<'ll, 'tcx>,
144    ptr_type: Ty<'tcx>,
145    pointee_type: Ty<'tcx>,
146    unique_type_id: UniqueTypeId<'tcx>,
147) -> DINodeCreationResult<'ll> {
148    // The debuginfo generated by this function is only valid if `ptr_type` is really just
149    // a (wide) pointer. Make sure it is not called for e.g. `Box<T, NonZSTAllocator>`.
150    match (&cx.size_and_align_of(ptr_type),
        &cx.size_and_align_of(Ty::new_mut_ptr(cx.tcx, pointee_type))) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(
151        cx.size_and_align_of(ptr_type),
152        cx.size_and_align_of(Ty::new_mut_ptr(cx.tcx, pointee_type))
153    );
154
155    let pointee_type_di_node = type_di_node(cx, pointee_type);
156
157    if let Some(di_node) =
        debug_context(cx).type_map.di_node_for_unique_id(unique_type_id) {
    return DINodeCreationResult::new(di_node, true);
};return_if_di_node_created_in_meantime!(cx, unique_type_id);
158
159    let data_layout = &cx.tcx.data_layout;
160    let pointer_size = data_layout.pointer_size();
161    let pointer_align = data_layout.pointer_align();
162    let ptr_type_debuginfo_name = compute_debuginfo_type_name(cx.tcx, ptr_type, true);
163
164    match wide_pointer_kind(cx, pointee_type) {
165        None => {
166            // This is a thin pointer. Create a regular pointer type and give it the correct name.
167            match (&(pointer_size, pointer_align.abi), &cx.size_and_align_of(ptr_type)) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::Some(format_args!("ptr_type={0}, pointee_type={1}",
                        ptr_type, pointee_type)));
        }
    }
};assert_eq!(
168                (pointer_size, pointer_align.abi),
169                cx.size_and_align_of(ptr_type),
170                "ptr_type={ptr_type}, pointee_type={pointee_type}",
171            );
172
173            let di_node = create_pointer_type(
174                cx,
175                pointee_type_di_node,
176                pointer_size,
177                pointer_align.abi,
178                &ptr_type_debuginfo_name,
179            );
180
181            DINodeCreationResult { di_node, already_stored_in_typemap: false }
182        }
183        Some(wide_pointer_kind) => {
184            type_map::build_type_with_children(
185                cx,
186                type_map::stub(
187                    cx,
188                    Stub::Struct,
189                    unique_type_id,
190                    &ptr_type_debuginfo_name,
191                    None,
192                    cx.size_and_align_of(ptr_type),
193                    NO_SCOPE_METADATA,
194                    DIFlags::FlagZero,
195                ),
196                |cx, owner| {
197                    // FIXME: If this wide pointer is a `Box` then we don't want to use its
198                    //        type layout and instead use the layout of the raw pointer inside
199                    //        of it.
200                    //        The proper way to handle this is to not treat Box as a pointer
201                    //        at all and instead emit regular struct debuginfo for it. We just
202                    //        need to make sure that we don't break existing debuginfo consumers
203                    //        by doing that (at least not without a warning period).
204                    let layout_type = if ptr_type.is_box() {
205                        // The assertion at the start of this function ensures we have a ZST
206                        // allocator. We'll make debuginfo "skip" all ZST allocators, not just the
207                        // default allocator.
208                        Ty::new_mut_ptr(cx.tcx, pointee_type)
209                    } else {
210                        ptr_type
211                    };
212
213                    let layout = cx.layout_of(layout_type);
214                    let addr_field = layout.field(cx, WIDE_PTR_ADDR);
215                    let extra_field = layout.field(cx, WIDE_PTR_EXTRA);
216
217                    let (addr_field_name, extra_field_name) = match wide_pointer_kind {
218                        WidePtrKind::Dyn => ("pointer", "vtable"),
219                        WidePtrKind::Slice => ("data_ptr", "length"),
220                    };
221
222                    match (&WIDE_PTR_ADDR, &0) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(WIDE_PTR_ADDR, 0);
223                    match (&WIDE_PTR_EXTRA, &1) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(WIDE_PTR_EXTRA, 1);
224
225                    // The data pointer type is a regular, thin pointer, regardless of whether this
226                    // is a slice or a trait object.
227                    let data_ptr_type_di_node = create_pointer_type(
228                        cx,
229                        pointee_type_di_node,
230                        addr_field.size,
231                        addr_field.align.abi,
232                        "",
233                    );
234
235                    {
    let count = 0usize + 1usize + 1usize;
    let mut vec = ::smallvec::SmallVec::new();
    if count <= vec.inline_size() {
        vec.push(build_field_di_node(cx, owner, addr_field_name, addr_field,
                layout.fields.offset(WIDE_PTR_ADDR), DIFlags::FlagZero,
                data_ptr_type_di_node, None));
        vec.push(build_field_di_node(cx, owner, extra_field_name, extra_field,
                layout.fields.offset(WIDE_PTR_EXTRA), DIFlags::FlagZero,
                type_di_node(cx, extra_field.ty), None));
        vec
    } else {
        ::smallvec::SmallVec::from_vec(<[_]>::into_vec(::alloc::boxed::box_new([build_field_di_node(cx,
                                owner, addr_field_name, addr_field,
                                layout.fields.offset(WIDE_PTR_ADDR), DIFlags::FlagZero,
                                data_ptr_type_di_node, None),
                            build_field_di_node(cx, owner, extra_field_name,
                                extra_field, layout.fields.offset(WIDE_PTR_EXTRA),
                                DIFlags::FlagZero, type_di_node(cx, extra_field.ty),
                                None)])))
    }
}smallvec![
236                        build_field_di_node(
237                            cx,
238                            owner,
239                            addr_field_name,
240                            addr_field,
241                            layout.fields.offset(WIDE_PTR_ADDR),
242                            DIFlags::FlagZero,
243                            data_ptr_type_di_node,
244                            None,
245                        ),
246                        build_field_di_node(
247                            cx,
248                            owner,
249                            extra_field_name,
250                            extra_field,
251                            layout.fields.offset(WIDE_PTR_EXTRA),
252                            DIFlags::FlagZero,
253                            type_di_node(cx, extra_field.ty),
254                            None,
255                        ),
256                    ]
257                },
258                NO_GENERICS,
259            )
260        }
261    }
262}
263
264fn build_subroutine_type_di_node<'ll, 'tcx>(
265    cx: &CodegenCx<'ll, 'tcx>,
266    unique_type_id: UniqueTypeId<'tcx>,
267) -> DINodeCreationResult<'ll> {
268    // It's possible to create a self-referential type in Rust by using 'impl trait':
269    //
270    // fn foo() -> impl Copy { foo }
271    //
272    // Unfortunately LLVM's API does not allow us to create recursive subroutine types.
273    // In order to work around that restriction we place a marker type in the type map,
274    // before creating the actual type. If the actual type is recursive, it will hit the
275    // marker type. So we end up with a type that looks like
276    //
277    // fn foo() -> <recursive_type>
278    //
279    // Once that is created, we replace the marker in the typemap with the actual type.
280    debug_context(cx)
281        .type_map
282        .unique_id_to_di_node
283        .borrow_mut()
284        .insert(unique_type_id, recursion_marker_type_di_node(cx));
285
286    let fn_ty = unique_type_id.expect_ty();
287    let signature =
288        cx.tcx.normalize_erasing_late_bound_regions(cx.typing_env(), fn_ty.fn_sig(cx.tcx));
289
290    let signature_di_nodes: SmallVec<_> = iter::once(
291        // return type
292        match signature.output().kind() {
293            ty::Tuple(tys) if tys.is_empty() => {
294                // this is a "void" function
295                None
296            }
297            _ => Some(type_di_node(cx, signature.output())),
298        },
299    )
300    .chain(
301        // regular arguments
302        signature.inputs().iter().map(|&argument_type| Some(type_di_node(cx, argument_type))),
303    )
304    .collect();
305
306    debug_context(cx).type_map.unique_id_to_di_node.borrow_mut().remove(&unique_type_id);
307
308    let fn_di_node = create_subroutine_type(cx, &signature_di_nodes[..]);
309
310    // This is actually a function pointer, so wrap it in pointer DI.
311    let name = compute_debuginfo_type_name(cx.tcx, fn_ty, false);
312    let (size, align) = match fn_ty.kind() {
313        ty::FnDef(..) => (Size::ZERO, Align::ONE),
314        ty::FnPtr(..) => {
315            (cx.tcx.data_layout.pointer_size(), cx.tcx.data_layout.pointer_align().abi)
316        }
317        _ => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
318    };
319    let di_node = create_pointer_type(cx, fn_di_node, size, align, &name);
320
321    DINodeCreationResult::new(di_node, false)
322}
323
324pub(super) fn create_subroutine_type<'ll>(
325    cx: &CodegenCx<'ll, '_>,
326    signature: &[Option<&'ll llvm::Metadata>],
327) -> &'ll DICompositeType {
328    unsafe {
329        llvm::LLVMDIBuilderCreateSubroutineType(
330            DIB(cx),
331            None, // ("File" is ignored and has no effect)
332            signature.as_ptr(),
333            signature.len() as c_uint,
334            DIFlags::FlagZero, // (default value)
335        )
336    }
337}
338
339fn create_pointer_type<'ll>(
340    cx: &CodegenCx<'ll, '_>,
341    pointee_ty: &'ll llvm::Metadata,
342    size: Size,
343    align: Align,
344    name: &str,
345) -> &'ll llvm::Metadata {
346    unsafe {
347        llvm::LLVMDIBuilderCreatePointerType(
348            DIB(cx),
349            pointee_ty,
350            size.bits(),
351            align.bits() as u32,
352            0, // Ignore DWARF address space.
353            name.as_ptr(),
354            name.len(),
355        )
356    }
357}
358
359/// Create debuginfo for `dyn SomeTrait` types. Currently these are empty structs
360/// we with the correct type name (e.g. "dyn SomeTrait<Foo, Item=u32> + Sync").
361fn build_dyn_type_di_node<'ll, 'tcx>(
362    cx: &CodegenCx<'ll, 'tcx>,
363    dyn_type: Ty<'tcx>,
364    unique_type_id: UniqueTypeId<'tcx>,
365) -> DINodeCreationResult<'ll> {
366    if let ty::Dynamic(..) = dyn_type.kind() {
367        let type_name = compute_debuginfo_type_name(cx.tcx, dyn_type, true);
368        type_map::build_type_with_children(
369            cx,
370            type_map::stub(
371                cx,
372                Stub::Struct,
373                unique_type_id,
374                &type_name,
375                None,
376                cx.size_and_align_of(dyn_type),
377                NO_SCOPE_METADATA,
378                DIFlags::FlagZero,
379            ),
380            |_, _| ::smallvec::SmallVec::new()smallvec![],
381            NO_GENERICS,
382        )
383    } else {
384        ::rustc_middle::util::bug::bug_fmt(format_args!("Only ty::Dynamic is valid for build_dyn_type_di_node(). Found {0:?} instead.",
        dyn_type))bug!(
385            "Only ty::Dynamic is valid for build_dyn_type_di_node(). Found {:?} instead.",
386            dyn_type
387        )
388    }
389}
390
391/// Create debuginfo for `[T]` and `str`. These are unsized.
392///
393/// NOTE: We currently emit just emit the debuginfo for the element type here
394/// (i.e. `T` for slices and `u8` for `str`), so that we end up with
395/// `*const T` for the `data_ptr` field of the corresponding wide-pointer
396/// debuginfo of `&[T]`.
397///
398/// It would be preferable and more accurate if we emitted a DIArray of T
399/// without an upper bound instead. That is, LLVM already supports emitting
400/// debuginfo of arrays of unknown size. But GDB currently seems to end up
401/// in an infinite loop when confronted with such a type.
402///
403/// As a side effect of the current encoding every instance of a type like
404/// `struct Foo { unsized_field: [u8] }` will look like
405/// `struct Foo { unsized_field: u8 }` in debuginfo. If the length of the
406/// slice is zero, then accessing `unsized_field` in the debugger would
407/// result in an out-of-bounds access.
408fn build_slice_type_di_node<'ll, 'tcx>(
409    cx: &CodegenCx<'ll, 'tcx>,
410    slice_type: Ty<'tcx>,
411    unique_type_id: UniqueTypeId<'tcx>,
412) -> DINodeCreationResult<'ll> {
413    let element_type = match slice_type.kind() {
414        ty::Slice(element_type) => *element_type,
415        ty::Str => cx.tcx.types.u8,
416        _ => {
417            ::rustc_middle::util::bug::bug_fmt(format_args!("Only ty::Slice is valid for build_slice_type_di_node(). Found {0:?} instead.",
        slice_type))bug!(
418                "Only ty::Slice is valid for build_slice_type_di_node(). Found {:?} instead.",
419                slice_type
420            )
421        }
422    };
423
424    let element_type_di_node = type_di_node(cx, element_type);
425    if let Some(di_node) =
        debug_context(cx).type_map.di_node_for_unique_id(unique_type_id) {
    return DINodeCreationResult::new(di_node, true);
};return_if_di_node_created_in_meantime!(cx, unique_type_id);
426    DINodeCreationResult { di_node: element_type_di_node, already_stored_in_typemap: false }
427}
428
429/// Get the debuginfo node for the given type.
430///
431/// This function will look up the debuginfo node in the TypeMap. If it can't find it, it
432/// will create the node by dispatching to the corresponding `build_*_di_node()` function.
433pub(crate) fn type_di_node<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>, t: Ty<'tcx>) -> &'ll DIType {
434    spanned_type_di_node(cx, t, DUMMY_SP)
435}
436
437pub(crate) fn spanned_type_di_node<'ll, 'tcx>(
438    cx: &CodegenCx<'ll, 'tcx>,
439    t: Ty<'tcx>,
440    span: Span,
441) -> &'ll DIType {
442    let unique_type_id = UniqueTypeId::for_ty(cx.tcx, t);
443
444    if let Some(existing_di_node) = debug_context(cx).type_map.di_node_for_unique_id(unique_type_id)
445    {
446        return existing_di_node;
447    }
448
449    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs:449",
                        "rustc_codegen_llvm::debuginfo::metadata",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs"),
                        ::tracing_core::__macro_support::Option::Some(449u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_codegen_llvm::debuginfo::metadata"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("type_di_node: {0:?} kind: {1:?}",
                                                    t, t.kind()) as &dyn Value))])
            });
    } else { ; }
};debug!("type_di_node: {:?} kind: {:?}", t, t.kind());
450
451    let DINodeCreationResult { di_node, already_stored_in_typemap } = match *t.kind() {
452        ty::Never | ty::Bool | ty::Char | ty::Int(_) | ty::Uint(_) | ty::Float(_) => {
453            build_basic_type_di_node(cx, t)
454        }
455        ty::Tuple(elements) if elements.is_empty() => build_basic_type_di_node(cx, t),
456        ty::Array(..) => build_fixed_size_array_di_node(cx, unique_type_id, t, span),
457        ty::Slice(_) | ty::Str => build_slice_type_di_node(cx, t, unique_type_id),
458        ty::Dynamic(..) => build_dyn_type_di_node(cx, t, unique_type_id),
459        ty::Foreign(..) => build_foreign_type_di_node(cx, t, unique_type_id),
460        ty::RawPtr(pointee_type, _) | ty::Ref(_, pointee_type, _) => {
461            build_pointer_or_reference_di_node(cx, t, pointee_type, unique_type_id)
462        }
463        // Some `Box` are newtyped pointers, make debuginfo aware of that.
464        // Only works if the allocator argument is a 1-ZST and hence irrelevant for layout
465        // (or if there is no allocator argument).
466        ty::Adt(def, args)
467            if def.is_box()
468                && args.get(1).is_none_or(|arg| cx.layout_of(arg.expect_ty()).is_1zst()) =>
469        {
470            build_pointer_or_reference_di_node(cx, t, t.expect_boxed_ty(), unique_type_id)
471        }
472        ty::FnDef(..) | ty::FnPtr(..) => build_subroutine_type_di_node(cx, unique_type_id),
473        ty::Closure(..) => build_closure_env_di_node(cx, unique_type_id),
474        ty::CoroutineClosure(..) => build_closure_env_di_node(cx, unique_type_id),
475        ty::Coroutine(..) => enums::build_coroutine_di_node(cx, unique_type_id),
476        ty::Adt(def, ..) => match def.adt_kind() {
477            AdtKind::Struct => build_struct_type_di_node(cx, unique_type_id, span),
478            AdtKind::Union => build_union_type_di_node(cx, unique_type_id, span),
479            AdtKind::Enum => enums::build_enum_type_di_node(cx, unique_type_id, span),
480        },
481        ty::Tuple(_) => build_tuple_type_di_node(cx, unique_type_id),
482        ty::Pat(base, _) => return type_di_node(cx, base),
483        // FIXME(unsafe_binders): impl debug info
484        ty::UnsafeBinder(_) => ::core::panicking::panic("not implemented")unimplemented!(),
485        ty::Alias(..)
486        | ty::Param(_)
487        | ty::Bound(..)
488        | ty::Infer(_)
489        | ty::Placeholder(_)
490        | ty::CoroutineWitness(..)
491        | ty::Error(_) => {
492            ::rustc_middle::util::bug::bug_fmt(format_args!("debuginfo: unexpected type in type_di_node(): {0:?}",
        t))bug!("debuginfo: unexpected type in type_di_node(): {:?}", t)
493        }
494    };
495
496    {
497        if already_stored_in_typemap {
498            // Make sure that we really do have a `TypeMap` entry for the unique type ID.
499            let di_node_for_uid =
500                match debug_context(cx).type_map.di_node_for_unique_id(unique_type_id) {
501                    Some(di_node) => di_node,
502                    None => {
503                        ::rustc_middle::util::bug::bug_fmt(format_args!("expected type debuginfo node for unique type ID \'{0:?}\' to already be in the `debuginfo::TypeMap` but it was not.",
        unique_type_id));bug!(
504                            "expected type debuginfo node for unique \
505                               type ID '{:?}' to already be in \
506                               the `debuginfo::TypeMap` but it \
507                               was not.",
508                            unique_type_id,
509                        );
510                    }
511                };
512
513            match (&(di_node_for_uid as *const _), &(di_node as *const _)) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(di_node_for_uid as *const _, di_node as *const _);
514        } else {
515            debug_context(cx).type_map.insert(unique_type_id, di_node);
516        }
517    }
518
519    di_node
520}
521
522// FIXME(mw): Cache this via a regular UniqueTypeId instead of an extra field in the debug context.
523fn recursion_marker_type_di_node<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>) -> &'ll DIType {
524    *debug_context(cx).recursion_marker_type.get_or_init(move || {
525        // The choice of type here is pretty arbitrary -
526        // anything reading the debuginfo for a recursive
527        // type is going to see *something* weird - the only
528        // question is what exactly it will see.
529        //
530        // FIXME: the name `<recur_type>` does not fit the naming scheme
531        //        of other types.
532        //
533        // FIXME: it might make sense to use an actual pointer type here
534        //        so that debuggers can show the address.
535        create_basic_type(
536            cx,
537            "<recur_type>",
538            cx.tcx.data_layout.pointer_size(),
539            dwarf_const::DW_ATE_unsigned,
540        )
541    })
542}
543
544fn hex_encode(data: &[u8]) -> String {
545    let mut hex_string = String::with_capacity(data.len() * 2);
546    for byte in data.iter() {
547        (&mut hex_string).write_fmt(format_args!("{0:02x}", byte))write!(&mut hex_string, "{byte:02x}").unwrap();
548    }
549    hex_string
550}
551
552pub(crate) fn file_metadata<'ll>(cx: &CodegenCx<'ll, '_>, source_file: &SourceFile) -> &'ll DIFile {
553    let cache_key = Some((source_file.stable_id, source_file.src_hash));
554    return debug_context(cx)
555        .created_files
556        .borrow_mut()
557        .entry(cache_key)
558        .or_insert_with(|| alloc_new_file_metadata(cx, source_file));
559
560    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() ||
            { false } {
        __tracing_attr_span =
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("alloc_new_file_metadata",
                                    "rustc_codegen_llvm::debuginfo::metadata",
                                    ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs"),
                                    ::tracing_core::__macro_support::Option::Some(560u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_codegen_llvm::debuginfo::metadata"),
                                    ::tracing_core::field::FieldSet::new(&[],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::LevelFilter::current() &&
                            { interest = __CALLSITE.interest(); !interest.is_never() }
                        &&
                        ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                            interest) {
                    let meta = __CALLSITE.metadata();
                    ::tracing::Span::new(meta,
                        &{ meta.fields().value_set(&[]) })
                } else {
                    let span =
                        ::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
                    {};
                    span
                }
            };
        __tracing_attr_guard = __tracing_attr_span.enter();
    }

    #[warn(clippy :: suspicious_else_formatting)]
    {

        #[allow(unknown_lints, unreachable_code, clippy ::
        diverging_sub_expression, clippy :: empty_loop, clippy ::
        let_unit_value, clippy :: let_with_type_underscore, clippy ::
        needless_return, clippy :: unreachable)]
        if false {
            let __tracing_attr_fake_return: &'ll DIFile = loop {};
            return __tracing_attr_fake_return;
        }
        {
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("event compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs:565",
                                    "rustc_codegen_llvm::debuginfo::metadata",
                                    ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs"),
                                    ::tracing_core::__macro_support::Option::Some(565u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_codegen_llvm::debuginfo::metadata"),
                                    ::tracing_core::field::FieldSet::new(&["source_file.name"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::EVENT)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let enabled =
                    ::tracing::Level::DEBUG <=
                                ::tracing::level_filters::STATIC_MAX_LEVEL &&
                            ::tracing::Level::DEBUG <=
                                ::tracing::level_filters::LevelFilter::current() &&
                        {
                            let interest = __CALLSITE.interest();
                            !interest.is_never() &&
                                ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                                    interest)
                        };
                if enabled {
                    (|value_set: ::tracing::field::ValueSet|
                                {
                                    let meta = __CALLSITE.metadata();
                                    ::tracing::Event::dispatch(meta, &value_set);
                                    ;
                                })({
                            #[allow(unused_imports)]
                            use ::tracing::field::{debug, display, Value};
                            let mut iter = __CALLSITE.metadata().fields().iter();
                            __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                ::tracing::__macro_support::Option::Some(&debug(&source_file.name)
                                                        as &dyn Value))])
                        });
                } else { ; }
            };
            let (directory, file_name) =
                match &source_file.name {
                    FileName::Real(filename) => {
                        let (working_directory, embeddable_name) =
                            filename.embeddable_name(RemapPathScopeComponents::DEBUGINFO);
                        {
                            use ::tracing::__macro_support::Callsite as _;
                            static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                                {
                                    static META: ::tracing::Metadata<'static> =
                                        {
                                            ::tracing_core::metadata::Metadata::new("event compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs:572",
                                                "rustc_codegen_llvm::debuginfo::metadata",
                                                ::tracing::Level::DEBUG,
                                                ::tracing_core::__macro_support::Option::Some("compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs"),
                                                ::tracing_core::__macro_support::Option::Some(572u32),
                                                ::tracing_core::__macro_support::Option::Some("rustc_codegen_llvm::debuginfo::metadata"),
                                                ::tracing_core::field::FieldSet::new(&["working_directory",
                                                                "embeddable_name"],
                                                    ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                                ::tracing::metadata::Kind::EVENT)
                                        };
                                    ::tracing::callsite::DefaultCallsite::new(&META)
                                };
                            let enabled =
                                ::tracing::Level::DEBUG <=
                                            ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                        ::tracing::Level::DEBUG <=
                                            ::tracing::level_filters::LevelFilter::current() &&
                                    {
                                        let interest = __CALLSITE.interest();
                                        !interest.is_never() &&
                                            ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                                                interest)
                                    };
                            if enabled {
                                (|value_set: ::tracing::field::ValueSet|
                                            {
                                                let meta = __CALLSITE.metadata();
                                                ::tracing::Event::dispatch(meta, &value_set);
                                                ;
                                            })({
                                        #[allow(unused_imports)]
                                        use ::tracing::field::{debug, display, Value};
                                        let mut iter = __CALLSITE.metadata().fields().iter();
                                        __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                            ::tracing::__macro_support::Option::Some(&debug(&working_directory)
                                                                    as &dyn Value)),
                                                        (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                            ::tracing::__macro_support::Option::Some(&debug(&embeddable_name)
                                                                    as &dyn Value))])
                                    });
                            } else { ; }
                        };
                        if let Ok(rel_path) =
                                embeddable_name.strip_prefix(working_directory) {
                            (working_directory.to_string_lossy(),
                                rel_path.to_string_lossy().into_owned())
                        } else {
                            ("".into(), embeddable_name.to_string_lossy().into_owned())
                        }
                    }
                    other => {
                        {
                            use ::tracing::__macro_support::Callsite as _;
                            static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                                {
                                    static META: ::tracing::Metadata<'static> =
                                        {
                                            ::tracing_core::metadata::Metadata::new("event compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs:597",
                                                "rustc_codegen_llvm::debuginfo::metadata",
                                                ::tracing::Level::DEBUG,
                                                ::tracing_core::__macro_support::Option::Some("compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs"),
                                                ::tracing_core::__macro_support::Option::Some(597u32),
                                                ::tracing_core::__macro_support::Option::Some("rustc_codegen_llvm::debuginfo::metadata"),
                                                ::tracing_core::field::FieldSet::new(&["other"],
                                                    ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                                ::tracing::metadata::Kind::EVENT)
                                        };
                                    ::tracing::callsite::DefaultCallsite::new(&META)
                                };
                            let enabled =
                                ::tracing::Level::DEBUG <=
                                            ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                        ::tracing::Level::DEBUG <=
                                            ::tracing::level_filters::LevelFilter::current() &&
                                    {
                                        let interest = __CALLSITE.interest();
                                        !interest.is_never() &&
                                            ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                                                interest)
                                    };
                            if enabled {
                                (|value_set: ::tracing::field::ValueSet|
                                            {
                                                let meta = __CALLSITE.metadata();
                                                ::tracing::Event::dispatch(meta, &value_set);
                                                ;
                                            })({
                                        #[allow(unused_imports)]
                                        use ::tracing::field::{debug, display, Value};
                                        let mut iter = __CALLSITE.metadata().fields().iter();
                                        __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                            ::tracing::__macro_support::Option::Some(&debug(&other) as
                                                                    &dyn Value))])
                                    });
                            } else { ; }
                        };
                        ("".into(),
                            other.display(RemapPathScopeComponents::DEBUGINFO).to_string())
                    }
                };
            let hash_kind =
                match source_file.src_hash.kind {
                    rustc_span::SourceFileHashAlgorithm::Md5 =>
                        llvm::ChecksumKind::MD5,
                    rustc_span::SourceFileHashAlgorithm::Sha1 =>
                        llvm::ChecksumKind::SHA1,
                    rustc_span::SourceFileHashAlgorithm::Sha256 =>
                        llvm::ChecksumKind::SHA256,
                    rustc_span::SourceFileHashAlgorithm::Blake3 =>
                        llvm::ChecksumKind::None,
                };
            let hash_value = hex_encode(source_file.src_hash.hash_bytes());
            let source =
                cx.sess().opts.unstable_opts.embed_source.then_some(()).and(source_file.src.as_ref());
            create_file(DIB(cx), &file_name, &directory, &hash_value,
                hash_kind, source)
        }
    }
}#[instrument(skip(cx, source_file), level = "debug")]
561    fn alloc_new_file_metadata<'ll>(
562        cx: &CodegenCx<'ll, '_>,
563        source_file: &SourceFile,
564    ) -> &'ll DIFile {
565        debug!(?source_file.name);
566
567        let (directory, file_name) = match &source_file.name {
568            FileName::Real(filename) => {
569                let (working_directory, embeddable_name) =
570                    filename.embeddable_name(RemapPathScopeComponents::DEBUGINFO);
571
572                debug!(?working_directory, ?embeddable_name);
573
574                if let Ok(rel_path) = embeddable_name.strip_prefix(working_directory) {
575                    // If the compiler's working directory (which also is the DW_AT_comp_dir of
576                    // the compilation unit) is a prefix of the path we are about to emit, then
577                    // only emit the part relative to the working directory. Because of path
578                    // remapping we sometimes see strange things here: `abs_path` might
579                    // actually look like a relative path (e.g.
580                    // `<crate-name-and-version>/src/lib.rs`), so if we emit it without taking
581                    // the working directory into account, downstream tooling will interpret it
582                    // as `<working-directory>/<crate-name-and-version>/src/lib.rs`, which
583                    // makes no sense. Usually in such cases the working directory will also be
584                    // remapped to `<crate-name-and-version>` or some other prefix of the path
585                    // we are remapping, so we end up with
586                    // `<crate-name-and-version>/<crate-name-and-version>/src/lib.rs`.
587                    //
588                    // By moving the working directory portion into the `directory` part of the
589                    // DIFile, we allow LLVM to emit just the relative path for DWARF, while
590                    // still emitting the correct absolute path for CodeView.
591                    (working_directory.to_string_lossy(), rel_path.to_string_lossy().into_owned())
592                } else {
593                    ("".into(), embeddable_name.to_string_lossy().into_owned())
594                }
595            }
596            other => {
597                debug!(?other);
598                ("".into(), other.display(RemapPathScopeComponents::DEBUGINFO).to_string())
599            }
600        };
601
602        let hash_kind = match source_file.src_hash.kind {
603            rustc_span::SourceFileHashAlgorithm::Md5 => llvm::ChecksumKind::MD5,
604            rustc_span::SourceFileHashAlgorithm::Sha1 => llvm::ChecksumKind::SHA1,
605            rustc_span::SourceFileHashAlgorithm::Sha256 => llvm::ChecksumKind::SHA256,
606            rustc_span::SourceFileHashAlgorithm::Blake3 => llvm::ChecksumKind::None,
607        };
608        let hash_value = hex_encode(source_file.src_hash.hash_bytes());
609
610        let source =
611            cx.sess().opts.unstable_opts.embed_source.then_some(()).and(source_file.src.as_ref());
612
613        create_file(DIB(cx), &file_name, &directory, &hash_value, hash_kind, source)
614    }
615}
616
617fn unknown_file_metadata<'ll>(cx: &CodegenCx<'ll, '_>) -> &'ll DIFile {
618    debug_context(cx).created_files.borrow_mut().entry(None).or_insert_with(|| {
619        create_file(DIB(cx), "<unknown>", "", "", llvm::ChecksumKind::None, None)
620    })
621}
622
623fn create_file<'ll>(
624    builder: &DIBuilder<'ll>,
625    file_name: &str,
626    directory: &str,
627    hash_value: &str,
628    hash_kind: llvm::ChecksumKind,
629    source: Option<&Arc<String>>,
630) -> &'ll DIFile {
631    unsafe {
632        llvm::LLVMRustDIBuilderCreateFile(
633            builder,
634            file_name.as_c_char_ptr(),
635            file_name.len(),
636            directory.as_c_char_ptr(),
637            directory.len(),
638            hash_kind,
639            hash_value.as_c_char_ptr(),
640            hash_value.len(),
641            source.map_or(ptr::null(), |x| x.as_c_char_ptr()),
642            source.map_or(0, |x| x.len()),
643        )
644    }
645}
646
647trait MsvcBasicName {
648    fn msvc_basic_name(self) -> &'static str;
649}
650
651impl MsvcBasicName for ty::IntTy {
652    fn msvc_basic_name(self) -> &'static str {
653        match self {
654            ty::IntTy::Isize => "ptrdiff_t",
655            ty::IntTy::I8 => "__int8",
656            ty::IntTy::I16 => "__int16",
657            ty::IntTy::I32 => "__int32",
658            ty::IntTy::I64 => "__int64",
659            ty::IntTy::I128 => "__int128",
660        }
661    }
662}
663
664impl MsvcBasicName for ty::UintTy {
665    fn msvc_basic_name(self) -> &'static str {
666        match self {
667            ty::UintTy::Usize => "size_t",
668            ty::UintTy::U8 => "unsigned __int8",
669            ty::UintTy::U16 => "unsigned __int16",
670            ty::UintTy::U32 => "unsigned __int32",
671            ty::UintTy::U64 => "unsigned __int64",
672            ty::UintTy::U128 => "unsigned __int128",
673        }
674    }
675}
676
677impl MsvcBasicName for ty::FloatTy {
678    fn msvc_basic_name(self) -> &'static str {
679        // FIXME(f128): `f128` has no MSVC representation. We could improve the debuginfo.
680        // See: <https://github.com/rust-lang/rust/issues/121837>
681        match self {
682            ty::FloatTy::F16 => {
683                ::rustc_middle::util::bug::bug_fmt(format_args!("`f16` should have been handled in `build_basic_type_di_node`"))bug!("`f16` should have been handled in `build_basic_type_di_node`")
684            }
685            ty::FloatTy::F32 => "float",
686            ty::FloatTy::F64 => "double",
687            ty::FloatTy::F128 => "fp128",
688        }
689    }
690}
691
692fn build_cpp_f16_di_node<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>) -> DINodeCreationResult<'ll> {
693    // MSVC has no native support for `f16`. Instead, emit `struct f16 { bits: u16 }` to allow the
694    // `f16`'s value to be displayed using a Natvis visualiser in `intrinsic.natvis`.
695    let float_ty = cx.tcx.types.f16;
696    let bits_ty = cx.tcx.types.u16;
697    let def_location = if cx.sess().opts.unstable_opts.debug_info_type_line_numbers {
698        match float_ty.kind() {
699            ty::Adt(def, _) => Some(file_metadata_from_def_id(cx, Some(def.did()))),
700            _ => None,
701        }
702    } else {
703        None
704    };
705    type_map::build_type_with_children(
706        cx,
707        type_map::stub(
708            cx,
709            Stub::Struct,
710            UniqueTypeId::for_ty(cx.tcx, float_ty),
711            "f16",
712            def_location,
713            cx.size_and_align_of(float_ty),
714            NO_SCOPE_METADATA,
715            DIFlags::FlagZero,
716        ),
717        // Fields:
718        |cx, float_di_node| {
719            let def_id = if cx.sess().opts.unstable_opts.debug_info_type_line_numbers {
720                match bits_ty.kind() {
721                    ty::Adt(def, _) => Some(def.did()),
722                    _ => None,
723                }
724            } else {
725                None
726            };
727            {
    let count = 0usize + 1usize;
    let mut vec = ::smallvec::SmallVec::new();
    if count <= vec.inline_size() {
        vec.push(build_field_di_node(cx, float_di_node, "bits",
                cx.layout_of(bits_ty), Size::ZERO, DIFlags::FlagZero,
                type_di_node(cx, bits_ty), def_id));
        vec
    } else {
        ::smallvec::SmallVec::from_vec(<[_]>::into_vec(::alloc::boxed::box_new([build_field_di_node(cx,
                                float_di_node, "bits", cx.layout_of(bits_ty), Size::ZERO,
                                DIFlags::FlagZero, type_di_node(cx, bits_ty), def_id)])))
    }
}smallvec![build_field_di_node(
728                cx,
729                float_di_node,
730                "bits",
731                cx.layout_of(bits_ty),
732                Size::ZERO,
733                DIFlags::FlagZero,
734                type_di_node(cx, bits_ty),
735                def_id,
736            )]
737        },
738        NO_GENERICS,
739    )
740}
741
742fn build_basic_type_di_node<'ll, 'tcx>(
743    cx: &CodegenCx<'ll, 'tcx>,
744    t: Ty<'tcx>,
745) -> DINodeCreationResult<'ll> {
746    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs:746",
                        "rustc_codegen_llvm::debuginfo::metadata",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs"),
                        ::tracing_core::__macro_support::Option::Some(746u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_codegen_llvm::debuginfo::metadata"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("build_basic_type_di_node: {0:?}",
                                                    t) as &dyn Value))])
            });
    } else { ; }
};debug!("build_basic_type_di_node: {:?}", t);
747
748    // When targeting MSVC, emit MSVC style type names for compatibility with
749    // .natvis visualizers (and perhaps other existing native debuggers?)
750    let cpp_like_debuginfo = cpp_like_debuginfo(cx.tcx);
751
752    use dwarf_const::{DW_ATE_UTF, DW_ATE_boolean, DW_ATE_float, DW_ATE_signed, DW_ATE_unsigned};
753
754    let (name, encoding) = match t.kind() {
755        ty::Never => ("!", DW_ATE_unsigned),
756        ty::Tuple(elements) if elements.is_empty() => {
757            if cpp_like_debuginfo {
758                return build_tuple_type_di_node(cx, UniqueTypeId::for_ty(cx.tcx, t));
759            } else {
760                ("()", DW_ATE_unsigned)
761            }
762        }
763        ty::Bool => ("bool", DW_ATE_boolean),
764        ty::Char => ("char", DW_ATE_UTF),
765        ty::Int(int_ty) if cpp_like_debuginfo => (int_ty.msvc_basic_name(), DW_ATE_signed),
766        ty::Uint(uint_ty) if cpp_like_debuginfo => (uint_ty.msvc_basic_name(), DW_ATE_unsigned),
767        ty::Float(ty::FloatTy::F16) if cpp_like_debuginfo => {
768            return build_cpp_f16_di_node(cx);
769        }
770        ty::Float(float_ty) if cpp_like_debuginfo => (float_ty.msvc_basic_name(), DW_ATE_float),
771        ty::Int(int_ty) => (int_ty.name_str(), DW_ATE_signed),
772        ty::Uint(uint_ty) => (uint_ty.name_str(), DW_ATE_unsigned),
773        ty::Float(float_ty) => (float_ty.name_str(), DW_ATE_float),
774        _ => ::rustc_middle::util::bug::bug_fmt(format_args!("debuginfo::build_basic_type_di_node - `t` is invalid type"))bug!("debuginfo::build_basic_type_di_node - `t` is invalid type"),
775    };
776
777    let ty_di_node = create_basic_type(cx, name, cx.size_of(t), encoding);
778
779    if !cpp_like_debuginfo {
780        return DINodeCreationResult::new(ty_di_node, false);
781    }
782
783    let typedef_name = match t.kind() {
784        ty::Int(int_ty) => int_ty.name_str(),
785        ty::Uint(uint_ty) => uint_ty.name_str(),
786        ty::Float(float_ty) => float_ty.name_str(),
787        _ => return DINodeCreationResult::new(ty_di_node, false),
788    };
789
790    let typedef_di_node = unsafe {
791        llvm::LLVMDIBuilderCreateTypedef(
792            DIB(cx),
793            ty_di_node,
794            typedef_name.as_ptr(),
795            typedef_name.len(),
796            unknown_file_metadata(cx),
797            0,    // (no line number)
798            None, // (no scope)
799            0u32, // (no alignment specified)
800        )
801    };
802
803    DINodeCreationResult::new(typedef_di_node, false)
804}
805
806fn create_basic_type<'ll, 'tcx>(
807    cx: &CodegenCx<'ll, 'tcx>,
808    name: &str,
809    size: Size,
810    encoding: u32,
811) -> &'ll DIBasicType {
812    unsafe {
813        llvm::LLVMDIBuilderCreateBasicType(
814            DIB(cx),
815            name.as_ptr(),
816            name.len(),
817            size.bits(),
818            encoding,
819            DIFlags::FlagZero,
820        )
821    }
822}
823
824fn build_foreign_type_di_node<'ll, 'tcx>(
825    cx: &CodegenCx<'ll, 'tcx>,
826    t: Ty<'tcx>,
827    unique_type_id: UniqueTypeId<'tcx>,
828) -> DINodeCreationResult<'ll> {
829    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs:829",
                        "rustc_codegen_llvm::debuginfo::metadata",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs"),
                        ::tracing_core::__macro_support::Option::Some(829u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_codegen_llvm::debuginfo::metadata"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("build_foreign_type_di_node: {0:?}",
                                                    t) as &dyn Value))])
            });
    } else { ; }
};debug!("build_foreign_type_di_node: {:?}", t);
830
831    let &ty::Foreign(def_id) = unique_type_id.expect_ty().kind() else {
832        ::rustc_middle::util::bug::bug_fmt(format_args!("build_foreign_type_di_node() called with unexpected type: {0:?}",
        unique_type_id.expect_ty()));bug!(
833            "build_foreign_type_di_node() called with unexpected type: {:?}",
834            unique_type_id.expect_ty()
835        );
836    };
837
838    build_type_with_children(
839        cx,
840        type_map::stub(
841            cx,
842            Stub::Struct,
843            unique_type_id,
844            &compute_debuginfo_type_name(cx.tcx, t, false),
845            None,
846            cx.size_and_align_of(t),
847            Some(get_namespace_for_item(cx, def_id)),
848            DIFlags::FlagZero,
849        ),
850        |_, _| ::smallvec::SmallVec::new()smallvec![],
851        NO_GENERICS,
852    )
853}
854
855pub(crate) fn build_compile_unit_di_node<'ll, 'tcx>(
856    tcx: TyCtxt<'tcx>,
857    codegen_unit_name: &str,
858    debug_context: &CodegenUnitDebugContext<'ll, 'tcx>,
859) -> &'ll DIDescriptor {
860    let mut name_in_debuginfo = tcx
861        .sess
862        .local_crate_source_file()
863        .map(|src| src.path(RemapPathScopeComponents::DEBUGINFO).to_path_buf())
864        .unwrap_or_else(|| PathBuf::from(tcx.crate_name(LOCAL_CRATE).as_str()));
865
866    // To avoid breaking split DWARF, we need to ensure that each codegen unit
867    // has a unique `DW_AT_name`. This is because there's a remote chance that
868    // different codegen units for the same module will have entirely
869    // identical DWARF entries for the purpose of the DWO ID, which would
870    // violate Appendix F ("Split Dwarf Object Files") of the DWARF 5
871    // specification. LLVM uses the algorithm specified in section 7.32 "Type
872    // Signature Computation" to compute the DWO ID, which does not include
873    // any fields that would distinguish compilation units. So we must embed
874    // the codegen unit name into the `DW_AT_name`. (Issue #88521.)
875    //
876    // Additionally, the OSX linker has an idiosyncrasy where it will ignore
877    // some debuginfo if multiple object files with the same `DW_AT_name` are
878    // linked together.
879    //
880    // As a workaround for these two issues, we generate unique names for each
881    // object file. Those do not correspond to an actual source file but that
882    // is harmless.
883    name_in_debuginfo.push("@");
884    name_in_debuginfo.push(codegen_unit_name);
885
886    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs:886",
                        "rustc_codegen_llvm::debuginfo::metadata",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs"),
                        ::tracing_core::__macro_support::Option::Some(886u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_codegen_llvm::debuginfo::metadata"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("build_compile_unit_di_node: {0:?}",
                                                    name_in_debuginfo) as &dyn Value))])
            });
    } else { ; }
};debug!("build_compile_unit_di_node: {:?}", name_in_debuginfo);
887    let rustc_producer = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("rustc version {0}",
                tcx.sess.cfg_version))
    })format!("rustc version {}", tcx.sess.cfg_version);
888    // FIXME(#41252) Remove "clang LLVM" if we can get GDB and LLVM to play nice.
889    let producer = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("clang LLVM ({0})", rustc_producer))
    })format!("clang LLVM ({rustc_producer})");
890
891    let name_in_debuginfo = name_in_debuginfo.to_string_lossy();
892    let work_dir = tcx.sess.psess.source_map().working_dir();
893    let output_filenames = tcx.output_filenames(());
894    let split_name = if tcx.sess.target_can_use_split_dwarf()
895        && let Some(f) = output_filenames.split_dwarf_path(
896            tcx.sess.split_debuginfo(),
897            tcx.sess.opts.unstable_opts.split_dwarf_kind,
898            codegen_unit_name,
899            tcx.sess.invocation_temp.as_deref(),
900        ) {
901        // We get a path relative to the working directory from split_dwarf_path
902        Some(tcx.sess.source_map().path_mapping().to_real_filename(work_dir, f))
903    } else {
904        None
905    };
906    let split_name = split_name
907        .as_ref()
908        .map(|f| f.path(RemapPathScopeComponents::DEBUGINFO).to_string_lossy())
909        .unwrap_or_default();
910    let work_dir = work_dir.path(RemapPathScopeComponents::DEBUGINFO).to_string_lossy();
911    let kind = DebugEmissionKind::from_generic(tcx.sess.opts.debuginfo);
912
913    let dwarf_version = tcx.sess.dwarf_version();
914    let is_dwarf_kind =
915        #[allow(non_exhaustive_omitted_patterns)] match tcx.sess.target.debuginfo_kind
    {
    DebuginfoKind::Dwarf | DebuginfoKind::DwarfDsym => true,
    _ => false,
}matches!(tcx.sess.target.debuginfo_kind, DebuginfoKind::Dwarf | DebuginfoKind::DwarfDsym);
916    // Don't emit `.debug_pubnames` and `.debug_pubtypes` on DWARFv4 or lower.
917    let debug_name_table_kind = if is_dwarf_kind && dwarf_version <= 4 {
918        DebugNameTableKind::None
919    } else {
920        DebugNameTableKind::Default
921    };
922
923    unsafe {
924        let compile_unit_file = create_file(
925            debug_context.builder.as_ref(),
926            &name_in_debuginfo,
927            &work_dir,
928            "",
929            llvm::ChecksumKind::None,
930            None,
931        );
932
933        let unit_metadata = llvm::LLVMRustDIBuilderCreateCompileUnit(
934            debug_context.builder.as_ref(),
935            dwarf_const::DW_LANG_Rust,
936            compile_unit_file,
937            producer.as_c_char_ptr(),
938            producer.len(),
939            tcx.sess.opts.optimize != config::OptLevel::No,
940            c"".as_ptr(),
941            0,
942            // NB: this doesn't actually have any perceptible effect, it seems. LLVM will instead
943            // put the path supplied to `MCSplitDwarfFile` into the debug info of the final
944            // output(s).
945            split_name.as_c_char_ptr(),
946            split_name.len(),
947            kind,
948            0,
949            tcx.sess.opts.unstable_opts.split_dwarf_inlining,
950            debug_name_table_kind,
951        );
952
953        return unit_metadata;
954    };
955}
956
957/// Creates a `DW_TAG_member` entry inside the DIE represented by the given `type_di_node`.
958fn build_field_di_node<'ll, 'tcx>(
959    cx: &CodegenCx<'ll, 'tcx>,
960    owner: &'ll DIScope,
961    name: &str,
962    layout: TyAndLayout<'tcx>,
963    offset: Size,
964    flags: DIFlags,
965    type_di_node: &'ll DIType,
966    def_id: Option<DefId>,
967) -> &'ll DIType {
968    let (file_metadata, line_number) = if cx.sess().opts.unstable_opts.debug_info_type_line_numbers
969    {
970        file_metadata_from_def_id(cx, def_id)
971    } else {
972        (unknown_file_metadata(cx), UNKNOWN_LINE_NUMBER)
973    };
974    create_member_type(
975        cx,
976        owner,
977        name,
978        file_metadata,
979        line_number,
980        layout,
981        offset,
982        flags,
983        type_di_node,
984    )
985}
986
987fn create_member_type<'ll, 'tcx>(
988    cx: &CodegenCx<'ll, 'tcx>,
989    owner: &'ll DIScope,
990    name: &str,
991    file_metadata: &'ll DIType,
992    line_number: u32,
993    layout: TyAndLayout<'tcx>,
994    offset: Size,
995    flags: DIFlags,
996    type_di_node: &'ll DIType,
997) -> &'ll DIType {
998    unsafe {
999        llvm::LLVMDIBuilderCreateMemberType(
1000            DIB(cx),
1001            owner,
1002            name.as_ptr(),
1003            name.len(),
1004            file_metadata,
1005            line_number,
1006            layout.size.bits(),
1007            layout.align.bits() as u32,
1008            offset.bits(),
1009            flags,
1010            type_di_node,
1011        )
1012    }
1013}
1014
1015/// Returns the `DIFlags` corresponding to the visibility of the item identified by `did`.
1016///
1017/// `DIFlags::Flag{Public,Protected,Private}` correspond to `DW_AT_accessibility`
1018/// (public/protected/private) aren't exactly right for Rust, but neither is `DW_AT_visibility`
1019/// (local/exported/qualified), and there's no way to set `DW_AT_visibility` in LLVM's API.
1020fn visibility_di_flags<'ll, 'tcx>(
1021    cx: &CodegenCx<'ll, 'tcx>,
1022    did: DefId,
1023    type_did: DefId,
1024) -> DIFlags {
1025    let parent_did = cx.tcx.parent(type_did);
1026    let visibility = cx.tcx.visibility(did);
1027    match visibility {
1028        Visibility::Public => DIFlags::FlagPublic,
1029        // Private fields have a restricted visibility of the module containing the type.
1030        Visibility::Restricted(did) if did == parent_did => DIFlags::FlagPrivate,
1031        // `pub(crate)`/`pub(super)` visibilities are any other restricted visibility.
1032        Visibility::Restricted(..) => DIFlags::FlagProtected,
1033    }
1034}
1035
1036/// Creates the debuginfo node for a Rust struct type. Maybe be a regular struct or a tuple-struct.
1037fn build_struct_type_di_node<'ll, 'tcx>(
1038    cx: &CodegenCx<'ll, 'tcx>,
1039    unique_type_id: UniqueTypeId<'tcx>,
1040    span: Span,
1041) -> DINodeCreationResult<'ll> {
1042    let struct_type = unique_type_id.expect_ty();
1043    let ty::Adt(adt_def, _) = struct_type.kind() else {
1044        ::rustc_middle::util::bug::bug_fmt(format_args!("build_struct_type_di_node() called with non-struct-type: {0:?}",
        struct_type));bug!("build_struct_type_di_node() called with non-struct-type: {:?}", struct_type);
1045    };
1046    if !adt_def.is_struct() {
    ::core::panicking::panic("assertion failed: adt_def.is_struct()")
};assert!(adt_def.is_struct());
1047    let containing_scope = get_namespace_for_item(cx, adt_def.did());
1048    let struct_type_and_layout = cx.spanned_layout_of(struct_type, span);
1049    let variant_def = adt_def.non_enum_variant();
1050    let def_location = if cx.sess().opts.unstable_opts.debug_info_type_line_numbers {
1051        Some(file_metadata_from_def_id(cx, Some(adt_def.did())))
1052    } else {
1053        None
1054    };
1055
1056    type_map::build_type_with_children(
1057        cx,
1058        type_map::stub(
1059            cx,
1060            Stub::Struct,
1061            unique_type_id,
1062            &compute_debuginfo_type_name(cx.tcx, struct_type, false),
1063            def_location,
1064            size_and_align_of(struct_type_and_layout),
1065            Some(containing_scope),
1066            visibility_di_flags(cx, adt_def.did(), adt_def.did()),
1067        ),
1068        // Fields:
1069        |cx, owner| {
1070            variant_def
1071                .fields
1072                .iter()
1073                .enumerate()
1074                .map(|(i, f)| {
1075                    let field_name = if variant_def.ctor_kind() == Some(CtorKind::Fn) {
1076                        // This is a tuple struct
1077                        tuple_field_name(i)
1078                    } else {
1079                        // This is struct with named fields
1080                        Cow::Borrowed(f.name.as_str())
1081                    };
1082                    let field_layout = struct_type_and_layout.field(cx, i);
1083                    let def_id = if cx.sess().opts.unstable_opts.debug_info_type_line_numbers {
1084                        Some(f.did)
1085                    } else {
1086                        None
1087                    };
1088                    build_field_di_node(
1089                        cx,
1090                        owner,
1091                        &field_name[..],
1092                        field_layout,
1093                        struct_type_and_layout.fields.offset(i),
1094                        visibility_di_flags(cx, f.did, adt_def.did()),
1095                        type_di_node(cx, field_layout.ty),
1096                        def_id,
1097                    )
1098                })
1099                .collect()
1100        },
1101        |cx| build_generic_type_param_di_nodes(cx, struct_type),
1102    )
1103}
1104
1105//=-----------------------------------------------------------------------------
1106// Tuples
1107//=-----------------------------------------------------------------------------
1108
1109/// Builds the DW_TAG_member debuginfo nodes for the upvars of a closure or coroutine.
1110/// For a coroutine, this will handle upvars shared by all states.
1111fn build_upvar_field_di_nodes<'ll, 'tcx>(
1112    cx: &CodegenCx<'ll, 'tcx>,
1113    closure_or_coroutine_ty: Ty<'tcx>,
1114    closure_or_coroutine_di_node: &'ll DIType,
1115) -> SmallVec<&'ll DIType> {
1116    let (&def_id, up_var_tys) = match closure_or_coroutine_ty.kind() {
1117        ty::Coroutine(def_id, args) => (def_id, args.as_coroutine().prefix_tys()),
1118        ty::Closure(def_id, args) => (def_id, args.as_closure().upvar_tys()),
1119        ty::CoroutineClosure(def_id, args) => (def_id, args.as_coroutine_closure().upvar_tys()),
1120        _ => {
1121            ::rustc_middle::util::bug::bug_fmt(format_args!("build_upvar_field_di_nodes() called with non-closure-or-coroutine-type: {0:?}",
        closure_or_coroutine_ty))bug!(
1122                "build_upvar_field_di_nodes() called with non-closure-or-coroutine-type: {:?}",
1123                closure_or_coroutine_ty
1124            )
1125        }
1126    };
1127
1128    if !up_var_tys.iter().all(|t|
                t == cx.tcx.normalize_erasing_regions(cx.typing_env(), t)) {
    ::core::panicking::panic("assertion failed: up_var_tys.iter().all(|t|\n        t == cx.tcx.normalize_erasing_regions(cx.typing_env(), t))")
};assert!(up_var_tys.iter().all(|t| t == cx.tcx.normalize_erasing_regions(cx.typing_env(), t)));
1129
1130    let capture_names = cx.tcx.closure_saved_names_of_captured_variables(def_id);
1131    let layout = cx.layout_of(closure_or_coroutine_ty);
1132
1133    up_var_tys
1134        .into_iter()
1135        .zip(capture_names.iter())
1136        .enumerate()
1137        .map(|(index, (up_var_ty, capture_name))| {
1138            build_field_di_node(
1139                cx,
1140                closure_or_coroutine_di_node,
1141                capture_name.as_str(),
1142                cx.layout_of(up_var_ty),
1143                layout.fields.offset(index),
1144                DIFlags::FlagZero,
1145                type_di_node(cx, up_var_ty),
1146                None,
1147            )
1148        })
1149        .collect()
1150}
1151
1152/// Builds the DW_TAG_structure_type debuginfo node for a Rust tuple type.
1153fn build_tuple_type_di_node<'ll, 'tcx>(
1154    cx: &CodegenCx<'ll, 'tcx>,
1155    unique_type_id: UniqueTypeId<'tcx>,
1156) -> DINodeCreationResult<'ll> {
1157    let tuple_type = unique_type_id.expect_ty();
1158    let &ty::Tuple(component_types) = tuple_type.kind() else {
1159        ::rustc_middle::util::bug::bug_fmt(format_args!("build_tuple_type_di_node() called with non-tuple-type: {0:?}",
        tuple_type))bug!("build_tuple_type_di_node() called with non-tuple-type: {:?}", tuple_type)
1160    };
1161
1162    let tuple_type_and_layout = cx.layout_of(tuple_type);
1163    let type_name = compute_debuginfo_type_name(cx.tcx, tuple_type, false);
1164
1165    type_map::build_type_with_children(
1166        cx,
1167        type_map::stub(
1168            cx,
1169            Stub::Struct,
1170            unique_type_id,
1171            &type_name,
1172            None,
1173            size_and_align_of(tuple_type_and_layout),
1174            NO_SCOPE_METADATA,
1175            DIFlags::FlagZero,
1176        ),
1177        // Fields:
1178        |cx, tuple_di_node| {
1179            component_types
1180                .into_iter()
1181                .enumerate()
1182                .map(|(index, component_type)| {
1183                    build_field_di_node(
1184                        cx,
1185                        tuple_di_node,
1186                        &tuple_field_name(index),
1187                        cx.layout_of(component_type),
1188                        tuple_type_and_layout.fields.offset(index),
1189                        DIFlags::FlagZero,
1190                        type_di_node(cx, component_type),
1191                        None,
1192                    )
1193                })
1194                .collect()
1195        },
1196        NO_GENERICS,
1197    )
1198}
1199
1200/// Builds the debuginfo node for a closure environment.
1201fn build_closure_env_di_node<'ll, 'tcx>(
1202    cx: &CodegenCx<'ll, 'tcx>,
1203    unique_type_id: UniqueTypeId<'tcx>,
1204) -> DINodeCreationResult<'ll> {
1205    let closure_env_type = unique_type_id.expect_ty();
1206    let &(ty::Closure(def_id, _) | ty::CoroutineClosure(def_id, _)) = closure_env_type.kind()
1207    else {
1208        ::rustc_middle::util::bug::bug_fmt(format_args!("build_closure_env_di_node() called with non-closure-type: {0:?}",
        closure_env_type))bug!("build_closure_env_di_node() called with non-closure-type: {:?}", closure_env_type)
1209    };
1210    let containing_scope = get_namespace_for_item(cx, def_id);
1211    let type_name = compute_debuginfo_type_name(cx.tcx, closure_env_type, false);
1212
1213    let def_location = if cx.sess().opts.unstable_opts.debug_info_type_line_numbers {
1214        Some(file_metadata_from_def_id(cx, Some(def_id)))
1215    } else {
1216        None
1217    };
1218
1219    type_map::build_type_with_children(
1220        cx,
1221        type_map::stub(
1222            cx,
1223            Stub::Struct,
1224            unique_type_id,
1225            &type_name,
1226            def_location,
1227            cx.size_and_align_of(closure_env_type),
1228            Some(containing_scope),
1229            DIFlags::FlagZero,
1230        ),
1231        // Fields:
1232        |cx, owner| build_upvar_field_di_nodes(cx, closure_env_type, owner),
1233        NO_GENERICS,
1234    )
1235}
1236
1237/// Build the debuginfo node for a Rust `union` type.
1238fn build_union_type_di_node<'ll, 'tcx>(
1239    cx: &CodegenCx<'ll, 'tcx>,
1240    unique_type_id: UniqueTypeId<'tcx>,
1241    span: Span,
1242) -> DINodeCreationResult<'ll> {
1243    let union_type = unique_type_id.expect_ty();
1244    let (union_def_id, variant_def) = match union_type.kind() {
1245        ty::Adt(def, _) => (def.did(), def.non_enum_variant()),
1246        _ => ::rustc_middle::util::bug::bug_fmt(format_args!("build_union_type_di_node on a non-ADT"))bug!("build_union_type_di_node on a non-ADT"),
1247    };
1248    let containing_scope = get_namespace_for_item(cx, union_def_id);
1249    let union_ty_and_layout = cx.spanned_layout_of(union_type, span);
1250    let type_name = compute_debuginfo_type_name(cx.tcx, union_type, false);
1251    let def_location = if cx.sess().opts.unstable_opts.debug_info_type_line_numbers {
1252        Some(file_metadata_from_def_id(cx, Some(union_def_id)))
1253    } else {
1254        None
1255    };
1256
1257    type_map::build_type_with_children(
1258        cx,
1259        type_map::stub(
1260            cx,
1261            Stub::Union,
1262            unique_type_id,
1263            &type_name,
1264            def_location,
1265            size_and_align_of(union_ty_and_layout),
1266            Some(containing_scope),
1267            DIFlags::FlagZero,
1268        ),
1269        // Fields:
1270        |cx, owner| {
1271            variant_def
1272                .fields
1273                .iter()
1274                .enumerate()
1275                .map(|(i, f)| {
1276                    let field_layout = union_ty_and_layout.field(cx, i);
1277                    let def_id = if cx.sess().opts.unstable_opts.debug_info_type_line_numbers {
1278                        Some(f.did)
1279                    } else {
1280                        None
1281                    };
1282                    build_field_di_node(
1283                        cx,
1284                        owner,
1285                        f.name.as_str(),
1286                        field_layout,
1287                        Size::ZERO,
1288                        DIFlags::FlagZero,
1289                        type_di_node(cx, field_layout.ty),
1290                        def_id,
1291                    )
1292                })
1293                .collect()
1294        },
1295        // Generics:
1296        |cx| build_generic_type_param_di_nodes(cx, union_type),
1297    )
1298}
1299
1300/// Computes the type parameters for a type, if any, for the given metadata.
1301fn build_generic_type_param_di_nodes<'ll, 'tcx>(
1302    cx: &CodegenCx<'ll, 'tcx>,
1303    ty: Ty<'tcx>,
1304) -> SmallVec<Option<&'ll DIType>> {
1305    if let ty::Adt(def, args) = *ty.kind() {
1306        if args.types().next().is_some() {
1307            let generics = cx.tcx.generics_of(def.did());
1308            let names = get_parameter_names(cx, generics);
1309            let template_params: SmallVec<_> = iter::zip(args, names)
1310                .filter_map(|(kind, name)| {
1311                    kind.as_type().map(|ty| {
1312                        let actual_type = cx.tcx.normalize_erasing_regions(cx.typing_env(), ty);
1313                        let actual_type_di_node = type_di_node(cx, actual_type);
1314                        Some(cx.create_template_type_parameter(name.as_str(), actual_type_di_node))
1315                    })
1316                })
1317                .collect();
1318
1319            return template_params;
1320        }
1321    }
1322
1323    return ::smallvec::SmallVec::new()smallvec![];
1324
1325    fn get_parameter_names(cx: &CodegenCx<'_, '_>, generics: &ty::Generics) -> Vec<Symbol> {
1326        let mut names = generics
1327            .parent
1328            .map_or_else(Vec::new, |def_id| get_parameter_names(cx, cx.tcx.generics_of(def_id)));
1329        names.extend(generics.own_params.iter().map(|param| param.name));
1330        names
1331    }
1332}
1333
1334/// Creates debug information for the given global variable.
1335///
1336/// Adds the created debuginfo nodes directly to the crate's IR.
1337pub(crate) fn build_global_var_di_node<'ll>(
1338    cx: &CodegenCx<'ll, '_>,
1339    def_id: DefId,
1340    global: &'ll Value,
1341) {
1342    if cx.dbg_cx.is_none() {
1343        return;
1344    }
1345
1346    // Only create type information if full debuginfo is enabled
1347    if cx.sess().opts.debuginfo != DebugInfo::Full {
1348        return;
1349    }
1350
1351    let tcx = cx.tcx;
1352
1353    // We may want to remove the namespace scope if we're in an extern block (see
1354    // https://github.com/rust-lang/rust/pull/46457#issuecomment-351750952).
1355    let var_scope = get_namespace_for_item(cx, def_id);
1356    let (file_metadata, line_number) = file_metadata_from_def_id(cx, Some(def_id));
1357
1358    let is_local_to_unit = is_node_local_to_unit(cx, def_id);
1359
1360    let DefKind::Static { nested, .. } = cx.tcx.def_kind(def_id) else { ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"))bug!() };
1361    if nested {
1362        return;
1363    }
1364    let variable_type = Instance::mono(cx.tcx, def_id).ty(cx.tcx, cx.typing_env());
1365    let type_di_node = type_di_node(cx, variable_type);
1366    let var_name = tcx.item_name(def_id);
1367    let var_name = var_name.as_str();
1368    let linkage_name = mangled_name_of_instance(cx, Instance::mono(tcx, def_id)).name;
1369    // When empty, linkage_name field is omitted,
1370    // which is what we want for no_mangle statics
1371    let linkage_name = if var_name == linkage_name { "" } else { linkage_name };
1372
1373    let global_align = cx.align_of(variable_type);
1374
1375    DIB(cx).create_static_variable(
1376        Some(var_scope),
1377        var_name,
1378        linkage_name,
1379        file_metadata,
1380        line_number,
1381        type_di_node,
1382        is_local_to_unit,
1383        global, // (value)
1384        None,   // (decl)
1385        Some(global_align),
1386    );
1387}
1388
1389/// Generates LLVM debuginfo for a vtable.
1390///
1391/// The vtable type looks like a struct with a field for each function pointer and super-trait
1392/// pointer it contains (plus the `size` and `align` fields).
1393///
1394/// Except for `size`, `align`, and `drop_in_place`, the field names don't try to mirror
1395/// the name of the method they implement. This can be implemented in the future once there
1396/// is a proper disambiguation scheme for dealing with methods from different traits that have
1397/// the same name.
1398fn build_vtable_type_di_node<'ll, 'tcx>(
1399    cx: &CodegenCx<'ll, 'tcx>,
1400    ty: Ty<'tcx>,
1401    poly_trait_ref: Option<ty::ExistentialTraitRef<'tcx>>,
1402) -> &'ll DIType {
1403    let tcx = cx.tcx;
1404
1405    let vtable_entries = if let Some(poly_trait_ref) = poly_trait_ref {
1406        let trait_ref = poly_trait_ref.with_self_ty(tcx, ty);
1407        let trait_ref = tcx.erase_and_anonymize_regions(trait_ref);
1408
1409        tcx.vtable_entries(trait_ref)
1410    } else {
1411        TyCtxt::COMMON_VTABLE_ENTRIES
1412    };
1413
1414    // All function pointers are described as opaque pointers. This could be improved in the future
1415    // by describing them as actual function pointers.
1416    let void_pointer_ty = Ty::new_imm_ptr(tcx, tcx.types.unit);
1417    let void_pointer_type_di_node = type_di_node(cx, void_pointer_ty);
1418    let usize_di_node = type_di_node(cx, tcx.types.usize);
1419    let pointer_layout = cx.layout_of(void_pointer_ty);
1420    let pointer_size = pointer_layout.size;
1421    let pointer_align = pointer_layout.align.abi;
1422    // If `usize` is not pointer-sized and -aligned then the size and alignment computations
1423    // for the vtable as a whole would be wrong. Let's make sure this holds even on weird
1424    // platforms.
1425    match (&cx.size_and_align_of(tcx.types.usize), &(pointer_size, pointer_align))
    {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(cx.size_and_align_of(tcx.types.usize), (pointer_size, pointer_align));
1426
1427    let vtable_type_name =
1428        compute_debuginfo_vtable_name(cx.tcx, ty, poly_trait_ref, VTableNameKind::Type);
1429    let unique_type_id = UniqueTypeId::for_vtable_ty(tcx, ty, poly_trait_ref);
1430    let size = pointer_size * vtable_entries.len() as u64;
1431
1432    // This gets mapped to a DW_AT_containing_type attribute which allows GDB to correlate
1433    // the vtable to the type it is for.
1434    let vtable_holder = type_di_node(cx, ty);
1435
1436    build_type_with_children(
1437        cx,
1438        type_map::stub(
1439            cx,
1440            Stub::VTableTy { vtable_holder },
1441            unique_type_id,
1442            &vtable_type_name,
1443            None,
1444            (size, pointer_align),
1445            NO_SCOPE_METADATA,
1446            DIFlags::FlagArtificial,
1447        ),
1448        |cx, vtable_type_di_node| {
1449            vtable_entries
1450                .iter()
1451                .enumerate()
1452                .filter_map(|(index, vtable_entry)| {
1453                    let (field_name, field_type_di_node) = match vtable_entry {
1454                        ty::VtblEntry::MetadataDropInPlace => {
1455                            ("drop_in_place".to_string(), void_pointer_type_di_node)
1456                        }
1457                        ty::VtblEntry::Method(_) => {
1458                            // Note: This code does not try to give a proper name to each method
1459                            //       because their might be multiple methods with the same name
1460                            //       (coming from different traits).
1461                            (::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("__method{0}", index))
    })format!("__method{index}"), void_pointer_type_di_node)
1462                        }
1463                        ty::VtblEntry::TraitVPtr(_) => {
1464                            (::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("__super_trait_ptr{0}", index))
    })format!("__super_trait_ptr{index}"), void_pointer_type_di_node)
1465                        }
1466                        ty::VtblEntry::MetadataAlign => ("align".to_string(), usize_di_node),
1467                        ty::VtblEntry::MetadataSize => ("size".to_string(), usize_di_node),
1468                        ty::VtblEntry::Vacant => return None,
1469                    };
1470
1471                    let field_offset = pointer_size * index as u64;
1472
1473                    Some(build_field_di_node(
1474                        cx,
1475                        vtable_type_di_node,
1476                        &field_name,
1477                        pointer_layout,
1478                        field_offset,
1479                        DIFlags::FlagZero,
1480                        field_type_di_node,
1481                        None,
1482                    ))
1483                })
1484                .collect()
1485        },
1486        NO_GENERICS,
1487    )
1488    .di_node
1489}
1490
1491/// Get the global variable for the vtable.
1492///
1493/// When using global variables, we may have created an addrspacecast to get a pointer to the
1494/// default address space if global variables are created in a different address space.
1495/// For modifying the vtable, we need the real global variable. This function accepts either a
1496/// global variable (which is simply returned), or an addrspacecast constant expression.
1497/// If the given value is an addrspacecast, the cast is removed and the global variable behind
1498/// the cast is returned.
1499fn find_vtable_behind_cast<'ll>(vtable: &'ll Value) -> &'ll Value {
1500    // The vtable is a global variable, which may be behind an addrspacecast.
1501    unsafe {
1502        if let Some(c) = llvm::LLVMIsAConstantExpr(vtable) {
1503            if llvm::LLVMGetConstOpcode(c) == llvm::Opcode::AddrSpaceCast {
1504                return llvm::LLVMGetOperand(c, 0).unwrap();
1505            }
1506        }
1507    }
1508    vtable
1509}
1510
1511pub(crate) fn apply_vcall_visibility_metadata<'ll, 'tcx>(
1512    cx: &CodegenCx<'ll, 'tcx>,
1513    ty: Ty<'tcx>,
1514    trait_ref: Option<ExistentialTraitRef<'tcx>>,
1515    vtable: &'ll Value,
1516) {
1517    // FIXME(flip1995): The virtual function elimination optimization only works with full LTO in
1518    // LLVM at the moment.
1519    if !cx.sess().opts.unstable_opts.virtual_function_elimination || cx.sess().lto() != Lto::Fat {
1520        return;
1521    }
1522
1523    enum VCallVisibility {
1524        Public = 0,
1525        LinkageUnit = 1,
1526        TranslationUnit = 2,
1527    }
1528
1529    let Some(trait_ref) = trait_ref else { return };
1530
1531    // Unwrap potential addrspacecast
1532    let vtable = find_vtable_behind_cast(vtable);
1533    let trait_ref_self = trait_ref.with_self_ty(cx.tcx, ty);
1534    let trait_def_id = trait_ref_self.def_id;
1535    let trait_vis = cx.tcx.visibility(trait_def_id);
1536
1537    let cgus = cx.sess().codegen_units().as_usize();
1538    let single_cgu = cgus == 1;
1539
1540    let lto = cx.sess().lto();
1541
1542    // Since LLVM requires full LTO for the virtual function elimination optimization to apply,
1543    // only the `Lto::Fat` cases are relevant currently.
1544    let vcall_visibility = match (lto, trait_vis, single_cgu) {
1545        // If there is not LTO and the visibility in public, we have to assume that the vtable can
1546        // be seen from anywhere. With multiple CGUs, the vtable is quasi-public.
1547        (Lto::No | Lto::ThinLocal, Visibility::Public, _)
1548        | (Lto::No, Visibility::Restricted(_), false) => VCallVisibility::Public,
1549        // With LTO and a quasi-public visibility, the usages of the functions of the vtable are
1550        // all known by the `LinkageUnit`.
1551        // FIXME: LLVM only supports this optimization for `Lto::Fat` currently. Once it also
1552        // supports `Lto::Thin` the `VCallVisibility` may have to be adjusted for those.
1553        (Lto::Fat | Lto::Thin, Visibility::Public, _)
1554        | (Lto::ThinLocal | Lto::Thin | Lto::Fat, Visibility::Restricted(_), false) => {
1555            VCallVisibility::LinkageUnit
1556        }
1557        // If there is only one CGU, private vtables can only be seen by that CGU/translation unit
1558        // and therefore we know of all usages of functions in the vtable.
1559        (_, Visibility::Restricted(_), true) => VCallVisibility::TranslationUnit,
1560    };
1561
1562    let trait_ref_typeid = typeid_for_trait_ref(cx.tcx, trait_ref);
1563    let typeid = cx.create_metadata(trait_ref_typeid.as_bytes());
1564
1565    let type_ = [llvm::LLVMValueAsMetadata(cx.const_usize(0)), typeid];
1566    cx.global_add_metadata_node(vtable, llvm::MD_type, &type_);
1567
1568    let vcall_visibility = [llvm::LLVMValueAsMetadata(cx.const_u64(vcall_visibility as u64))];
1569    cx.global_set_metadata_node(vtable, llvm::MD_vcall_visibility, &vcall_visibility);
1570}
1571
1572/// Creates debug information for the given vtable, which is for the
1573/// given type.
1574///
1575/// Adds the created metadata nodes directly to the crate's IR.
1576pub(crate) fn create_vtable_di_node<'ll, 'tcx>(
1577    cx: &CodegenCx<'ll, 'tcx>,
1578    ty: Ty<'tcx>,
1579    poly_trait_ref: Option<ty::ExistentialTraitRef<'tcx>>,
1580    vtable: &'ll Value,
1581) {
1582    if cx.dbg_cx.is_none() {
1583        return;
1584    }
1585
1586    // Only create type information if full debuginfo is enabled
1587    if cx.sess().opts.debuginfo != DebugInfo::Full {
1588        return;
1589    }
1590
1591    // Unwrap potential addrspacecast
1592    let vtable = find_vtable_behind_cast(vtable);
1593
1594    // When full debuginfo is enabled, we want to try and prevent vtables from being
1595    // merged. Otherwise debuggers will have a hard time mapping from dyn pointer
1596    // to concrete type.
1597    llvm::set_unnamed_address(vtable, llvm::UnnamedAddr::No);
1598
1599    let vtable_name =
1600        compute_debuginfo_vtable_name(cx.tcx, ty, poly_trait_ref, VTableNameKind::GlobalVariable);
1601    let vtable_type_di_node = build_vtable_type_di_node(cx, ty, poly_trait_ref);
1602
1603    DIB(cx).create_static_variable(
1604        NO_SCOPE_METADATA,
1605        &vtable_name,
1606        "", // (linkage_name)
1607        unknown_file_metadata(cx),
1608        UNKNOWN_LINE_NUMBER,
1609        vtable_type_di_node,
1610        true,   // (is_local_to_unit)
1611        vtable, // (value)
1612        None,   // (decl)
1613        None::<Align>,
1614    );
1615}
1616
1617/// Creates an "extension" of an existing `DIScope` into another file.
1618pub(crate) fn extend_scope_to_file<'ll>(
1619    cx: &CodegenCx<'ll, '_>,
1620    scope_metadata: &'ll DIScope,
1621    file: &SourceFile,
1622) -> &'ll DILexicalBlock {
1623    let file_metadata = file_metadata(cx, file);
1624    unsafe {
1625        llvm::LLVMDIBuilderCreateLexicalBlockFile(
1626            DIB(cx),
1627            scope_metadata,
1628            file_metadata,
1629            /* Discriminator (default) */ 0u32,
1630        )
1631    }
1632}
1633
1634fn tuple_field_name(field_index: usize) -> Cow<'static, str> {
1635    const TUPLE_FIELD_NAMES: [&'static str; 16] = [
1636        "__0", "__1", "__2", "__3", "__4", "__5", "__6", "__7", "__8", "__9", "__10", "__11",
1637        "__12", "__13", "__14", "__15",
1638    ];
1639    TUPLE_FIELD_NAMES
1640        .get(field_index)
1641        .map(|s| Cow::from(*s))
1642        .unwrap_or_else(|| Cow::from(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("__{0}", field_index))
    })format!("__{field_index}")))
1643}
1644
1645pub(crate) type DefinitionLocation<'ll> = (&'ll DIFile, c_uint);
1646
1647pub(crate) fn file_metadata_from_def_id<'ll>(
1648    cx: &CodegenCx<'ll, '_>,
1649    def_id: Option<DefId>,
1650) -> DefinitionLocation<'ll> {
1651    if let Some(def_id) = def_id
1652        && let span = hygiene::walk_chain_collapsed(cx.tcx.def_span(def_id), DUMMY_SP)
1653        && !span.is_dummy()
1654    {
1655        let loc = cx.lookup_debug_loc(span.lo());
1656        (file_metadata(cx, &loc.file), loc.line)
1657    } else {
1658        (unknown_file_metadata(cx), UNKNOWN_LINE_NUMBER)
1659    }
1660}