rustc_middle/mir/interpret/
queries.rs

1use rustc_hir::def::DefKind;
2use rustc_hir::def_id::DefId;
3use rustc_session::lint;
4use rustc_span::{DUMMY_SP, Span};
5use tracing::{debug, instrument};
6
7use super::{
8    ErrorHandled, EvalToAllocationRawResult, EvalToConstValueResult, EvalToValTreeResult, GlobalId,
9    ReportedErrorInfo,
10};
11use crate::mir;
12use crate::ty::{self, GenericArgs, TyCtxt, TypeVisitableExt};
13
14impl<'tcx> TyCtxt<'tcx> {
15    /// Evaluates a constant without providing any generic parameters. This is useful to evaluate consts
16    /// that can't take any generic arguments like const items or enum discriminants. If a
17    /// generic parameter is used within the constant `ErrorHandled::TooGeneric` will be returned.
18    #[instrument(skip(self), level = "debug")]
19    pub fn const_eval_poly(self, def_id: DefId) -> EvalToConstValueResult<'tcx> {
20        // In some situations def_id will have generic parameters within scope, but they aren't allowed
21        // to be used. So we can't use `Instance::mono`, instead we feed unresolved generic parameters
22        // into `const_eval` which will return `ErrorHandled::TooGeneric` if any of them are
23        // encountered.
24        let args = GenericArgs::identity_for_item(self, def_id);
25        let instance = ty::Instance::new_raw(def_id, args);
26        let cid = GlobalId { instance, promoted: None };
27        let typing_env = ty::TypingEnv::post_analysis(self, def_id);
28        self.const_eval_global_id(typing_env, cid, DUMMY_SP)
29    }
30
31    /// Evaluates a constant without providing any generic parameters. This is useful to evaluate consts
32    /// that can't take any generic arguments like const items or enum discriminants. If a
33    /// generic parameter is used within the constant `ErrorHandled::TooGeneric` will be returned.
34    #[instrument(skip(self), level = "debug")]
35    pub fn const_eval_poly_to_alloc(self, def_id: DefId) -> EvalToAllocationRawResult<'tcx> {
36        // In some situations def_id will have generic parameters within scope, but they aren't allowed
37        // to be used. So we can't use `Instance::mono`, instead we feed unresolved generic parameters
38        // into `const_eval` which will return `ErrorHandled::TooGeneric` if any of them are
39        // encountered.
40        let args = GenericArgs::identity_for_item(self, def_id);
41        let instance = ty::Instance::new_raw(def_id, args);
42        let cid = GlobalId { instance, promoted: None };
43        let typing_env = ty::TypingEnv::post_analysis(self, def_id);
44        let inputs = self.erase_regions(typing_env.as_query_input(cid));
45        self.eval_to_allocation_raw(inputs)
46    }
47
48    /// Resolves and evaluates a constant.
49    ///
50    /// The constant can be located on a trait like `<A as B>::C`, in which case the given
51    /// generic parameters and environment are used to resolve the constant. Alternatively if the
52    /// constant has generic parameters in scope the generic parameters are used to evaluate the value of
53    /// the constant. For example in `fn foo<T>() { let _ = [0; bar::<T>()]; }` the repeat count
54    /// constant `bar::<T>()` requires a instantiation for `T`, if the instantiation for `T` is still
55    /// too generic for the constant to be evaluated then `Err(ErrorHandled::TooGeneric)` is
56    /// returned.
57    #[instrument(level = "debug", skip(self))]
58    pub fn const_eval_resolve(
59        self,
60        typing_env: ty::TypingEnv<'tcx>,
61        ct: mir::UnevaluatedConst<'tcx>,
62        span: Span,
63    ) -> EvalToConstValueResult<'tcx> {
64        // Cannot resolve `Unevaluated` constants that contain inference
65        // variables. We reject those here since `resolve`
66        // would fail otherwise.
67        //
68        // When trying to evaluate constants containing inference variables,
69        // use `Infcx::const_eval_resolve` instead.
70        if ct.args.has_non_region_infer() {
71            bug!("did not expect inference variables here");
72        }
73
74        // FIXME: maybe have a separate version for resolving mir::UnevaluatedConst?
75        match ty::Instance::try_resolve(self, typing_env, ct.def, ct.args) {
76            Ok(Some(instance)) => {
77                let cid = GlobalId { instance, promoted: ct.promoted };
78                self.const_eval_global_id(typing_env, cid, span)
79            }
80            // For errors during resolution, we deliberately do not point at the usage site of the constant,
81            // since for these errors the place the constant is used shouldn't matter.
82            Ok(None) => Err(ErrorHandled::TooGeneric(DUMMY_SP)),
83            Err(err) => {
84                Err(ErrorHandled::Reported(ReportedErrorInfo::non_const_eval_error(err), DUMMY_SP))
85            }
86        }
87    }
88
89    #[instrument(level = "debug", skip(self))]
90    pub fn const_eval_resolve_for_typeck(
91        self,
92        typing_env: ty::TypingEnv<'tcx>,
93        ct: ty::UnevaluatedConst<'tcx>,
94        span: Span,
95    ) -> EvalToValTreeResult<'tcx> {
96        // Cannot resolve `Unevaluated` constants that contain inference
97        // variables. We reject those here since `resolve`
98        // would fail otherwise.
99        //
100        // When trying to evaluate constants containing inference variables,
101        // use `Infcx::const_eval_resolve` instead.
102        if ct.args.has_non_region_infer() {
103            bug!("did not expect inference variables here");
104        }
105
106        match ty::Instance::try_resolve(self, typing_env, ct.def, ct.args) {
107            Ok(Some(instance)) => {
108                let cid = GlobalId { instance, promoted: None };
109                self.const_eval_global_id_for_typeck(typing_env, cid, span).inspect(|_| {
110                    // We are emitting the lint here instead of in `is_const_evaluatable`
111                    // as we normalize obligations before checking them, and normalization
112                    // uses this function to evaluate this constant.
113                    //
114                    // @lcnr believes that successfully evaluating even though there are
115                    // used generic parameters is a bug of evaluation, so checking for it
116                    // here does feel somewhat sensible.
117                    if !self.features().generic_const_exprs()
118                        && ct.args.has_non_region_param()
119                        // We only FCW for anon consts as repeat expr counts with anon consts are the only place
120                        // that we have a back compat hack for. We don't need to check this is a const argument
121                        // as only anon consts as const args should get evaluated "for the type system".
122                        //
123                        // If we don't *only* FCW anon consts we can wind up incorrectly FCW'ing uses of assoc
124                        // consts in pattern positions. #140447
125                        && self.def_kind(instance.def_id()) == DefKind::AnonConst
126                    {
127                        let mir_body = self.mir_for_ctfe(instance.def_id());
128                        if mir_body.is_polymorphic {
129                            let Some(local_def_id) = ct.def.as_local() else { return };
130                            self.node_span_lint(
131                                lint::builtin::CONST_EVALUATABLE_UNCHECKED,
132                                self.local_def_id_to_hir_id(local_def_id),
133                                self.def_span(ct.def),
134                                |lint| { lint.primary_message("cannot use constants which depend on generic parameters in types"); },
135                            )
136                        }
137                    }
138                })
139            }
140            // For errors during resolution, we deliberately do not point at the usage site of the constant,
141            // since for these errors the place the constant is used shouldn't matter.
142            Ok(None) => Err(ErrorHandled::TooGeneric(DUMMY_SP)),
143            Err(err) => {
144                Err(ErrorHandled::Reported(ReportedErrorInfo::non_const_eval_error(err), DUMMY_SP))
145            }
146        }
147    }
148
149    pub fn const_eval_instance(
150        self,
151        typing_env: ty::TypingEnv<'tcx>,
152        instance: ty::Instance<'tcx>,
153        span: Span,
154    ) -> EvalToConstValueResult<'tcx> {
155        self.const_eval_global_id(typing_env, GlobalId { instance, promoted: None }, span)
156    }
157
158    /// Evaluate a constant to a `ConstValue`.
159    #[instrument(skip(self), level = "debug")]
160    pub fn const_eval_global_id(
161        self,
162        typing_env: ty::TypingEnv<'tcx>,
163        cid: GlobalId<'tcx>,
164        span: Span,
165    ) -> EvalToConstValueResult<'tcx> {
166        // Const-eval shouldn't depend on lifetimes at all, so we can erase them, which should
167        // improve caching of queries.
168        let inputs =
169            self.erase_regions(typing_env.with_post_analysis_normalized(self).as_query_input(cid));
170        if !span.is_dummy() {
171            // The query doesn't know where it is being invoked, so we need to fix the span.
172            self.at(span).eval_to_const_value_raw(inputs).map_err(|e| e.with_span(span))
173        } else {
174            self.eval_to_const_value_raw(inputs)
175        }
176    }
177
178    /// Evaluate a constant to a type-level constant.
179    #[instrument(skip(self), level = "debug")]
180    pub fn const_eval_global_id_for_typeck(
181        self,
182        typing_env: ty::TypingEnv<'tcx>,
183        cid: GlobalId<'tcx>,
184        span: Span,
185    ) -> EvalToValTreeResult<'tcx> {
186        // Const-eval shouldn't depend on lifetimes at all, so we can erase them, which should
187        // improve caching of queries.
188        let inputs =
189            self.erase_regions(typing_env.with_post_analysis_normalized(self).as_query_input(cid));
190        debug!(?inputs);
191        if !span.is_dummy() {
192            // The query doesn't know where it is being invoked, so we need to fix the span.
193            self.at(span).eval_to_valtree(inputs).map_err(|e| e.with_span(span))
194        } else {
195            self.eval_to_valtree(inputs)
196        }
197    }
198}