rustc_hir_analysis/collect/
generics_of.rs

1use std::assert_matches::assert_matches;
2use std::ops::ControlFlow;
3
4use rustc_hir::def::DefKind;
5use rustc_hir::def_id::LocalDefId;
6use rustc_hir::intravisit::{self, Visitor, VisitorExt};
7use rustc_hir::{self as hir, AmbigArg, GenericParamKind, HirId, Node};
8use rustc_middle::span_bug;
9use rustc_middle::ty::{self, TyCtxt};
10use rustc_session::lint;
11use rustc_span::{Span, Symbol, kw};
12use tracing::{debug, instrument};
13
14use crate::delegation::inherit_generics_for_delegation_item;
15use crate::middle::resolve_bound_vars as rbv;
16
17#[instrument(level = "debug", skip(tcx), ret)]
18pub(super) fn generics_of(tcx: TyCtxt<'_>, def_id: LocalDefId) -> ty::Generics {
19    use rustc_hir::*;
20
21    // For an RPITIT, synthesize generics which are equal to the opaque's generics
22    // and parent fn's generics compressed into one list.
23    if let Some(ty::ImplTraitInTraitData::Trait { fn_def_id, opaque_def_id }) =
24        tcx.opt_rpitit_info(def_id.to_def_id())
25    {
26        debug!("RPITIT fn_def_id={fn_def_id:?} opaque_def_id={opaque_def_id:?}");
27        let trait_def_id = tcx.parent(fn_def_id);
28        let opaque_ty_generics = tcx.generics_of(opaque_def_id);
29        let opaque_ty_parent_count = opaque_ty_generics.parent_count;
30        let mut own_params = opaque_ty_generics.own_params.clone();
31
32        let parent_generics = tcx.generics_of(trait_def_id);
33        let parent_count = parent_generics.parent_count + parent_generics.own_params.len();
34
35        let mut trait_fn_params = tcx.generics_of(fn_def_id).own_params.clone();
36
37        for param in &mut own_params {
38            param.index = param.index + parent_count as u32 + trait_fn_params.len() as u32
39                - opaque_ty_parent_count as u32;
40        }
41
42        trait_fn_params.extend(own_params);
43        own_params = trait_fn_params;
44
45        let param_def_id_to_index =
46            own_params.iter().map(|param| (param.def_id, param.index)).collect();
47
48        return ty::Generics {
49            parent: Some(trait_def_id),
50            parent_count,
51            own_params,
52            param_def_id_to_index,
53            has_self: opaque_ty_generics.has_self,
54            has_late_bound_regions: opaque_ty_generics.has_late_bound_regions,
55        };
56    }
57
58    let hir_id = tcx.local_def_id_to_hir_id(def_id);
59
60    let node = tcx.hir_node(hir_id);
61    if let Some(sig) = node.fn_sig()
62        && let Some(sig_id) = sig.decl.opt_delegation_sig_id()
63    {
64        return inherit_generics_for_delegation_item(tcx, def_id, sig_id);
65    }
66
67    let parent_def_id = match node {
68        Node::ImplItem(_)
69        | Node::TraitItem(_)
70        | Node::Variant(_)
71        | Node::Ctor(..)
72        | Node::Field(_) => {
73            let parent_id = tcx.hir_get_parent_item(hir_id);
74            Some(parent_id.to_def_id())
75        }
76        // FIXME(#43408) always enable this once `lazy_normalization` is
77        // stable enough and does not need a feature gate anymore.
78        Node::AnonConst(_) => {
79            let parent_did = tcx.parent(def_id.to_def_id());
80            debug!(?parent_did);
81
82            let mut in_param_ty = false;
83            for (_parent, node) in tcx.hir_parent_iter(hir_id) {
84                if let Some(generics) = node.generics() {
85                    let mut visitor = AnonConstInParamTyDetector { in_param_ty: false, ct: hir_id };
86
87                    in_param_ty = visitor.visit_generics(generics).is_break();
88                    break;
89                }
90            }
91
92            match tcx.anon_const_kind(def_id) {
93                // Stable: anon consts are not able to use any generic parameters...
94                ty::AnonConstKind::MCG => None,
95                // we provide generics to repeat expr counts as a backwards compatibility hack. #76200
96                ty::AnonConstKind::RepeatExprCount => Some(parent_did),
97
98                // Even GCE anon const should not be allowed to use generic parameters as it would be
99                // trivially forward declared uses once desugared. E.g. `const N: [u8; ANON::<N>]`.
100                //
101                // We could potentially mirror the hack done for defaults of generic parameters but
102                // this case just doesn't come up much compared to `const N: u32 = ...`. Long term the
103                // hack for defaulted parameters should be removed eventually anyway.
104                ty::AnonConstKind::GCE if in_param_ty => None,
105                // GCE anon consts as a default for a generic parameter should have their provided generics
106                // "truncated" up to whatever generic parameter this anon const is within the default of.
107                //
108                // FIXME(generic_const_exprs): This only handles `const N: usize = /*defid*/` but not type
109                // parameter defaults, e.g. `T = Foo</*defid*/>`.
110                ty::AnonConstKind::GCE
111                    if let Some(param_id) =
112                        tcx.hir_opt_const_param_default_param_def_id(hir_id) =>
113                {
114                    // If the def_id we are calling generics_of on is an anon ct default i.e:
115                    //
116                    // struct Foo<const N: usize = { .. }>;
117                    //        ^^^       ^          ^^^^^^ def id of this anon const
118                    //        ^         ^ param_id
119                    //        ^ parent_def_id
120                    //
121                    // then we only want to return generics for params to the left of `N`. If we don't do that we
122                    // end up with that const looking like: `ty::ConstKind::Unevaluated(def_id, args: [N#0])`.
123                    //
124                    // This causes ICEs (#86580) when building the args for Foo in `fn foo() -> Foo { .. }` as
125                    // we instantiate the defaults with the partially built args when we build the args. Instantiating
126                    // the `N#0` on the unevaluated const indexes into the empty args we're in the process of building.
127                    //
128                    // We fix this by having this function return the parent's generics ourselves and truncating the
129                    // generics to only include non-forward declared params (with the exception of the `Self` ty)
130                    //
131                    // For the above code example that means we want `args: []`
132                    // For the following struct def we want `args: [N#0]` when generics_of is called on
133                    // the def id of the `{ N + 1 }` anon const
134                    // struct Foo<const N: usize, const M: usize = { N + 1 }>;
135                    //
136                    // This has some implications for how we get the predicates available to the anon const
137                    // see `explicit_predicates_of` for more information on this
138                    let generics = tcx.generics_of(parent_did);
139                    let param_def_idx = generics.param_def_id_to_index[&param_id.to_def_id()];
140                    // In the above example this would be .params[..N#0]
141                    let own_params = generics.params_to(param_def_idx as usize, tcx).to_owned();
142                    let param_def_id_to_index =
143                        own_params.iter().map(|param| (param.def_id, param.index)).collect();
144
145                    return ty::Generics {
146                        // we set the parent of these generics to be our parent's parent so that we
147                        // dont end up with args: [N, M, N] for the const default on a struct like this:
148                        // struct Foo<const N: usize, const M: usize = { ... }>;
149                        parent: generics.parent,
150                        parent_count: generics.parent_count,
151                        own_params,
152                        param_def_id_to_index,
153                        has_self: generics.has_self,
154                        has_late_bound_regions: generics.has_late_bound_regions,
155                    };
156                }
157                ty::AnonConstKind::GCE => Some(parent_did),
158
159                // Field defaults are allowed to use generic parameters, e.g. `field: u32 = /*defid: N + 1*/`
160                ty::AnonConstKind::NonTypeSystem
161                    if matches!(tcx.parent_hir_node(hir_id), Node::TyPat(_) | Node::Field(_)) =>
162                {
163                    Some(parent_did)
164                }
165                // Default to no generic parameters for other kinds of anon consts
166                ty::AnonConstKind::NonTypeSystem => None,
167            }
168        }
169        Node::ConstBlock(_)
170        | Node::Expr(&hir::Expr { kind: hir::ExprKind::Closure { .. }, .. }) => {
171            Some(tcx.typeck_root_def_id(def_id.to_def_id()))
172        }
173        Node::OpaqueTy(&hir::OpaqueTy {
174            origin:
175                hir::OpaqueTyOrigin::FnReturn { parent: fn_def_id, in_trait_or_impl }
176                | hir::OpaqueTyOrigin::AsyncFn { parent: fn_def_id, in_trait_or_impl },
177            ..
178        }) => {
179            if in_trait_or_impl.is_some() {
180                assert_matches!(tcx.def_kind(fn_def_id), DefKind::AssocFn);
181            } else {
182                assert_matches!(tcx.def_kind(fn_def_id), DefKind::AssocFn | DefKind::Fn);
183            }
184            Some(fn_def_id.to_def_id())
185        }
186        Node::OpaqueTy(&hir::OpaqueTy {
187            origin: hir::OpaqueTyOrigin::TyAlias { parent, in_assoc_ty },
188            ..
189        }) => {
190            if in_assoc_ty {
191                assert_matches!(tcx.def_kind(parent), DefKind::AssocTy);
192            } else {
193                assert_matches!(tcx.def_kind(parent), DefKind::TyAlias);
194            }
195            debug!("generics_of: parent of opaque ty {:?} is {:?}", def_id, parent);
196            // Opaque types are always nested within another item, and
197            // inherit the generics of the item.
198            Some(parent.to_def_id())
199        }
200
201        // All of these nodes have no parent from which to inherit generics.
202        Node::Item(_) | Node::ForeignItem(_) => None,
203
204        // Params don't really have generics, but we use it when instantiating their value paths.
205        Node::GenericParam(_) => None,
206
207        Node::Synthetic => span_bug!(
208            tcx.def_span(def_id),
209            "synthetic HIR should have its `generics_of` explicitly fed"
210        ),
211
212        _ => span_bug!(tcx.def_span(def_id), "generics_of: unexpected node kind {node:?}"),
213    };
214
215    // Add in the self type parameter.
216    let opt_self = if let Node::Item(item) = node
217        && let ItemKind::Trait(..) | ItemKind::TraitAlias(..) = item.kind
218    {
219        // Something of a hack: We reuse the node ID of the trait for the self type parameter.
220        Some(ty::GenericParamDef {
221            index: 0,
222            name: kw::SelfUpper,
223            def_id: def_id.to_def_id(),
224            pure_wrt_drop: false,
225            kind: ty::GenericParamDefKind::Type { has_default: false, synthetic: false },
226        })
227    } else {
228        None
229    };
230
231    let param_default_policy = param_default_policy(node);
232    let hir_generics = node.generics().unwrap_or(hir::Generics::empty());
233    let has_self = opt_self.is_some();
234    let mut parent_has_self = false;
235    let mut own_start = has_self as u32;
236    let parent_count = parent_def_id.map_or(0, |def_id| {
237        let generics = tcx.generics_of(def_id);
238        assert!(!has_self);
239        parent_has_self = generics.has_self;
240        own_start = generics.count() as u32;
241        generics.parent_count + generics.own_params.len()
242    });
243
244    let mut own_params: Vec<_> = Vec::with_capacity(hir_generics.params.len() + has_self as usize);
245
246    if let Some(opt_self) = opt_self {
247        own_params.push(opt_self);
248    }
249
250    let early_lifetimes = super::early_bound_lifetimes_from_generics(tcx, hir_generics);
251    own_params.extend(early_lifetimes.enumerate().map(|(i, param)| ty::GenericParamDef {
252        name: param.name.ident().name,
253        index: own_start + i as u32,
254        def_id: param.def_id.to_def_id(),
255        pure_wrt_drop: param.pure_wrt_drop,
256        kind: ty::GenericParamDefKind::Lifetime,
257    }));
258
259    // Now create the real type and const parameters.
260    let type_start = own_start - has_self as u32 + own_params.len() as u32;
261    let mut i: u32 = 0;
262    let mut next_index = || {
263        let prev = i;
264        i += 1;
265        prev + type_start
266    };
267
268    own_params.extend(hir_generics.params.iter().filter_map(|param| {
269        const MESSAGE: &str = "defaults for generic parameters are not allowed here";
270        let kind = match param.kind {
271            GenericParamKind::Lifetime { .. } => return None,
272            GenericParamKind::Type { default, synthetic } => {
273                if default.is_some() {
274                    match param_default_policy.expect("no policy for generic param default") {
275                        ParamDefaultPolicy::Allowed => {}
276                        ParamDefaultPolicy::FutureCompatForbidden => {
277                            tcx.node_span_lint(
278                                lint::builtin::INVALID_TYPE_PARAM_DEFAULT,
279                                param.hir_id,
280                                param.span,
281                                |lint| {
282                                    lint.primary_message(MESSAGE);
283                                },
284                            );
285                        }
286                        ParamDefaultPolicy::Forbidden => {
287                            tcx.dcx().span_err(param.span, MESSAGE);
288                        }
289                    }
290                }
291
292                ty::GenericParamDefKind::Type { has_default: default.is_some(), synthetic }
293            }
294            GenericParamKind::Const { ty: _, default } => {
295                if default.is_some() {
296                    match param_default_policy.expect("no policy for generic param default") {
297                        ParamDefaultPolicy::Allowed => {}
298                        ParamDefaultPolicy::FutureCompatForbidden
299                        | ParamDefaultPolicy::Forbidden => {
300                            tcx.dcx().span_err(param.span, MESSAGE);
301                        }
302                    }
303                }
304
305                ty::GenericParamDefKind::Const { has_default: default.is_some() }
306            }
307        };
308        Some(ty::GenericParamDef {
309            index: next_index(),
310            name: param.name.ident().name,
311            def_id: param.def_id.to_def_id(),
312            pure_wrt_drop: param.pure_wrt_drop,
313            kind,
314        })
315    }));
316
317    // provide junk type parameter defs - the only place that
318    // cares about anything but the length is instantiation,
319    // and we don't do that for closures.
320    if let Node::Expr(&hir::Expr {
321        kind: hir::ExprKind::Closure(hir::Closure { kind, .. }), ..
322    }) = node
323    {
324        // See `ClosureArgsParts`, `CoroutineArgsParts`, and `CoroutineClosureArgsParts`
325        // for info on the usage of each of these fields.
326        let dummy_args = match kind {
327            ClosureKind::Closure => &["<closure_kind>", "<closure_signature>", "<upvars>"][..],
328            ClosureKind::Coroutine(_) => {
329                &["<coroutine_kind>", "<resume_ty>", "<yield_ty>", "<return_ty>", "<upvars>"][..]
330            }
331            ClosureKind::CoroutineClosure(_) => &[
332                "<closure_kind>",
333                "<closure_signature_parts>",
334                "<upvars>",
335                "<bound_captures_by_ref>",
336            ][..],
337        };
338
339        own_params.extend(dummy_args.iter().map(|&arg| ty::GenericParamDef {
340            index: next_index(),
341            name: Symbol::intern(arg),
342            def_id: def_id.to_def_id(),
343            pure_wrt_drop: false,
344            kind: ty::GenericParamDefKind::Type { has_default: false, synthetic: false },
345        }));
346    }
347
348    // provide junk type parameter defs for const blocks.
349    if let Node::ConstBlock(_) = node {
350        own_params.push(ty::GenericParamDef {
351            index: next_index(),
352            name: rustc_span::sym::const_ty_placeholder,
353            def_id: def_id.to_def_id(),
354            pure_wrt_drop: false,
355            kind: ty::GenericParamDefKind::Type { has_default: false, synthetic: false },
356        });
357    }
358
359    if let Node::OpaqueTy(&hir::OpaqueTy { .. }) = node {
360        assert!(own_params.is_empty());
361
362        let lifetimes = tcx.opaque_captured_lifetimes(def_id);
363        debug!(?lifetimes);
364
365        own_params.extend(lifetimes.iter().map(|&(_, param)| ty::GenericParamDef {
366            name: tcx.item_name(param.to_def_id()),
367            index: next_index(),
368            def_id: param.to_def_id(),
369            pure_wrt_drop: false,
370            kind: ty::GenericParamDefKind::Lifetime,
371        }))
372    }
373
374    let param_def_id_to_index =
375        own_params.iter().map(|param| (param.def_id, param.index)).collect();
376
377    ty::Generics {
378        parent: parent_def_id,
379        parent_count,
380        own_params,
381        param_def_id_to_index,
382        has_self: has_self || parent_has_self,
383        has_late_bound_regions: has_late_bound_regions(tcx, node),
384    }
385}
386
387#[derive(Clone, Copy)]
388enum ParamDefaultPolicy {
389    Allowed,
390    /// Tracked in <https://github.com/rust-lang/rust/issues/36887>.
391    FutureCompatForbidden,
392    Forbidden,
393}
394
395fn param_default_policy(node: Node<'_>) -> Option<ParamDefaultPolicy> {
396    use rustc_hir::*;
397
398    Some(match node {
399        Node::Item(item) => match item.kind {
400            ItemKind::Trait(..)
401            | ItemKind::TraitAlias(..)
402            | ItemKind::TyAlias(..)
403            | ItemKind::Enum(..)
404            | ItemKind::Struct(..)
405            | ItemKind::Union(..) => ParamDefaultPolicy::Allowed,
406            ItemKind::Fn { .. } | ItemKind::Impl(_) => ParamDefaultPolicy::FutureCompatForbidden,
407            // Re. GCI, we're not bound by backward compatibility.
408            ItemKind::Const(..) => ParamDefaultPolicy::Forbidden,
409            _ => return None,
410        },
411        Node::TraitItem(item) => match item.kind {
412            // Re. GATs and GACs (generic_const_items), we're not bound by backward compatibility.
413            TraitItemKind::Const(..) | TraitItemKind::Type(..) => ParamDefaultPolicy::Forbidden,
414            TraitItemKind::Fn(..) => ParamDefaultPolicy::FutureCompatForbidden,
415        },
416        Node::ImplItem(item) => match item.kind {
417            // Re. GATs and GACs (generic_const_items), we're not bound by backward compatibility.
418            ImplItemKind::Const(..) | ImplItemKind::Type(..) => ParamDefaultPolicy::Forbidden,
419            ImplItemKind::Fn(..) => ParamDefaultPolicy::FutureCompatForbidden,
420        },
421        // Generic params are (semantically) invalid on foreign items. Still, for maximum forward
422        // compatibility, let's hard-reject defaults on them.
423        Node::ForeignItem(_) => ParamDefaultPolicy::Forbidden,
424        Node::OpaqueTy(..) => ParamDefaultPolicy::Allowed,
425        _ => return None,
426    })
427}
428
429fn has_late_bound_regions<'tcx>(tcx: TyCtxt<'tcx>, node: Node<'tcx>) -> Option<Span> {
430    struct LateBoundRegionsDetector<'tcx> {
431        tcx: TyCtxt<'tcx>,
432        outer_index: ty::DebruijnIndex,
433    }
434
435    impl<'tcx> Visitor<'tcx> for LateBoundRegionsDetector<'tcx> {
436        type Result = ControlFlow<Span>;
437        fn visit_ty(&mut self, ty: &'tcx hir::Ty<'tcx, AmbigArg>) -> ControlFlow<Span> {
438            match ty.kind {
439                hir::TyKind::FnPtr(..) => {
440                    self.outer_index.shift_in(1);
441                    let res = intravisit::walk_ty(self, ty);
442                    self.outer_index.shift_out(1);
443                    res
444                }
445                hir::TyKind::UnsafeBinder(_) => {
446                    self.outer_index.shift_in(1);
447                    let res = intravisit::walk_ty(self, ty);
448                    self.outer_index.shift_out(1);
449                    res
450                }
451                _ => intravisit::walk_ty(self, ty),
452            }
453        }
454
455        fn visit_poly_trait_ref(&mut self, tr: &'tcx hir::PolyTraitRef<'tcx>) -> ControlFlow<Span> {
456            self.outer_index.shift_in(1);
457            let res = intravisit::walk_poly_trait_ref(self, tr);
458            self.outer_index.shift_out(1);
459            res
460        }
461
462        fn visit_lifetime(&mut self, lt: &'tcx hir::Lifetime) -> ControlFlow<Span> {
463            match self.tcx.named_bound_var(lt.hir_id) {
464                Some(rbv::ResolvedArg::StaticLifetime | rbv::ResolvedArg::EarlyBound(..)) => {
465                    ControlFlow::Continue(())
466                }
467                Some(rbv::ResolvedArg::LateBound(debruijn, _, _))
468                    if debruijn < self.outer_index =>
469                {
470                    ControlFlow::Continue(())
471                }
472                Some(
473                    rbv::ResolvedArg::LateBound(..)
474                    | rbv::ResolvedArg::Free(..)
475                    | rbv::ResolvedArg::Error(_),
476                )
477                | None => ControlFlow::Break(lt.ident.span),
478            }
479        }
480    }
481
482    fn has_late_bound_regions<'tcx>(
483        tcx: TyCtxt<'tcx>,
484        generics: &'tcx hir::Generics<'tcx>,
485        decl: &'tcx hir::FnDecl<'tcx>,
486    ) -> Option<Span> {
487        let mut visitor = LateBoundRegionsDetector { tcx, outer_index: ty::INNERMOST };
488        for param in generics.params {
489            if let GenericParamKind::Lifetime { .. } = param.kind {
490                if tcx.is_late_bound(param.hir_id) {
491                    return Some(param.span);
492                }
493            }
494        }
495        visitor.visit_fn_decl(decl).break_value()
496    }
497
498    let decl = node.fn_decl()?;
499    let generics = node.generics()?;
500    has_late_bound_regions(tcx, generics, decl)
501}
502
503struct AnonConstInParamTyDetector {
504    in_param_ty: bool,
505    ct: HirId,
506}
507
508impl<'v> Visitor<'v> for AnonConstInParamTyDetector {
509    type Result = ControlFlow<()>;
510
511    fn visit_generic_param(&mut self, p: &'v hir::GenericParam<'v>) -> Self::Result {
512        if let GenericParamKind::Const { ty, default: _ } = p.kind {
513            let prev = self.in_param_ty;
514            self.in_param_ty = true;
515            let res = self.visit_ty_unambig(ty);
516            self.in_param_ty = prev;
517            res
518        } else {
519            ControlFlow::Continue(())
520        }
521    }
522
523    fn visit_anon_const(&mut self, c: &'v hir::AnonConst) -> Self::Result {
524        if self.in_param_ty && self.ct == c.hir_id {
525            return ControlFlow::Break(());
526        }
527        intravisit::walk_anon_const(self, c)
528    }
529}