1use rustc_hir::def_id::{CRATE_DEF_ID, LocalDefId};
2use rustc_infer::infer::TyCtxtInferExt;
3use rustc_infer::traits::{Obligation, ObligationCause};
4use rustc_middle::mir::interpret::{AllocInit, Allocation, GlobalAlloc, InterpResult, Pointer};
5use rustc_middle::ty::layout::TyAndLayout;
6use rustc_middle::ty::{PolyExistentialPredicate, Ty, TyCtxt, TypeVisitable, TypeVisitableExt};
7use rustc_middle::{mir, span_bug, ty};
8use rustc_trait_selection::traits::ObligationCtxt;
9use tracing::debug;
1011use super::{InterpCx, MPlaceTy, MemoryKind, interp_ok, throw_inval};
12use crate::const_eval::{CompileTimeInterpCx, CompileTimeMachine, InterpretationResult};
13use crate::interpret::Machine;
1415/// Checks if a type implements predicates.
16/// Calls `ensure_monomorphic_enough` on `ty` and `trait_ty` for you.
17pub(crate) fn type_implements_dyn_trait<'tcx, M: Machine<'tcx>>(
18 ecx: &mut InterpCx<'tcx, M>,
19 ty: Ty<'tcx>,
20 trait_ty: Ty<'tcx>,
21) -> InterpResult<'tcx, (bool, &'tcx ty::List<ty::PolyExistentialPredicate<'tcx>>)> {
22ensure_monomorphic_enough(ecx.tcx.tcx, ty)?;
23ensure_monomorphic_enough(ecx.tcx.tcx, trait_ty)?;
2425let ty::Dynamic(preds, _) = trait_ty.kind() else {
26::rustc_middle::util::bug::span_bug_fmt(ecx.find_closest_untracked_caller_location(),
format_args!("Invalid type provided to type_implements_predicates. U must be dyn Trait, got {0}.",
trait_ty));span_bug!(
27ecx.find_closest_untracked_caller_location(),
28"Invalid type provided to type_implements_predicates. U must be dyn Trait, got {trait_ty}."
29);
30 };
3132let (infcx, param_env) = ecx.tcx.infer_ctxt().build_with_typing_env(ecx.typing_env);
3334let ocx = ObligationCtxt::new(&infcx);
35ocx.register_obligations(preds.iter().map(|pred: PolyExistentialPredicate<'_>| {
36let pred = pred.with_self_ty(ecx.tcx.tcx, ty);
37// Lifetimes can only be 'static because of the bound on T
38let pred = rustc_middle::ty::fold_regions(ecx.tcx.tcx, pred, |r, _| {
39if r == ecx.tcx.tcx.lifetimes.re_erased { ecx.tcx.tcx.lifetimes.re_static } else { r }
40 });
41 Obligation::new(ecx.tcx.tcx, ObligationCause::dummy(), param_env, pred)
42 }));
43let type_impls_trait = ocx.evaluate_obligations_error_on_ambiguity().is_empty();
44// Since `assumed_wf_tys=[]` the choice of LocalDefId is irrelevant, so using the "default"
45let regions_are_valid = ocx.resolve_regions(CRATE_DEF_ID, param_env, []).is_empty();
4647interp_ok((regions_are_valid && type_impls_trait, preds))
48}
4950/// Checks whether a type contains generic parameters which must be instantiated.
51///
52/// In case it does, returns a `TooGeneric` const eval error.
53pub(crate) fn ensure_monomorphic_enough<'tcx, T>(_tcx: TyCtxt<'tcx>, ty: T) -> InterpResult<'tcx>
54where
55T: TypeVisitable<TyCtxt<'tcx>>,
56{
57{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_const_eval/src/interpret/util.rs:57",
"rustc_const_eval::interpret::util",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/util.rs"),
::tracing_core::__macro_support::Option::Some(57u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::util"),
::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!("ensure_monomorphic_enough: ty={0:?}",
ty) as &dyn Value))])
});
} else { ; }
};debug!("ensure_monomorphic_enough: ty={:?}", ty);
58if ty.has_param() {
59do yeet ::rustc_middle::mir::interpret::InterpErrorKind::InvalidProgram(::rustc_middle::mir::interpret::InvalidProgramInfo::TooGeneric);throw_inval!(TooGeneric);
60 }
61interp_ok(())
62}
6364impl<'tcx> InterpretationResult<'tcx> for mir::interpret::ConstAllocation<'tcx> {
65fn make_result(
66 mplace: MPlaceTy<'tcx>,
67 ecx: &mut InterpCx<'tcx, CompileTimeMachine<'tcx>>,
68 ) -> Self {
69let alloc_id = mplace.ptr().provenance.unwrap().alloc_id();
70let alloc = ecx.memory.alloc_map.swap_remove(&alloc_id).unwrap().1;
71ecx.tcx.mk_const_alloc(alloc)
72 }
73}
7475pub(crate) fn create_static_alloc<'tcx>(
76 ecx: &mut CompileTimeInterpCx<'tcx>,
77 static_def_id: LocalDefId,
78 layout: TyAndLayout<'tcx>,
79) -> InterpResult<'tcx, MPlaceTy<'tcx>> {
80// Inherit size and align from the `GlobalAlloc::Static` so we can avoid duplicating
81 // the alignment attribute logic.
82let (size, align) =
83 GlobalAlloc::Static(static_def_id.into()).size_and_align(*ecx.tcx, ecx.typing_env);
84match (&size, &layout.size) {
(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!(size, layout.size);
85if !(align >= layout.align.abi) {
::core::panicking::panic("assertion failed: align >= layout.align.abi")
};assert!(align >= layout.align.abi);
8687let alloc = Allocation::try_new(size, align, AllocInit::Uninit, ())?;
88let alloc_id = ecx.tcx.reserve_and_set_static_alloc(static_def_id.into());
89match (&ecx.machine.static_root_ids, &None) {
(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!(ecx.machine.static_root_ids, None);
90ecx.machine.static_root_ids = Some((alloc_id, static_def_id));
91if !ecx.memory.alloc_map.insert(alloc_id,
(MemoryKind::Stack, alloc)).is_none() {
::core::panicking::panic("assertion failed: ecx.memory.alloc_map.insert(alloc_id, (MemoryKind::Stack, alloc)).is_none()")
};assert!(ecx.memory.alloc_map.insert(alloc_id, (MemoryKind::Stack, alloc)).is_none());
92interp_ok(ecx.ptr_to_mplace(Pointer::from(alloc_id).into(), layout))
93}
9495/// A marker trait returned by [crate::interpret::Machine::enter_trace_span], identifying either a
96/// real [tracing::span::EnteredSpan] in case tracing is enabled, or the dummy type `()` when
97/// tracing is disabled. Also see [crate::enter_trace_span!] below.
98pub trait EnteredTraceSpan {
99/// Allows executing an alternative function when tracing is disabled. Useful for example if you
100 /// want to open a trace span when tracing is enabled, and alternatively just log a line when
101 /// tracing is disabled.
102fn or_if_tracing_disabled(self, f: impl FnOnce()) -> Self;
103}
104impl EnteredTraceSpanfor () {
105fn or_if_tracing_disabled(self, f: impl FnOnce()) -> Self {
106f(); // tracing is disabled, execute the function
107self108 }
109}
110impl EnteredTraceSpanfor tracing::span::EnteredSpan {
111fn or_if_tracing_disabled(self, _f: impl FnOnce()) -> Self {
112self// tracing is enabled, don't execute anything
113}
114}
115116/// Shorthand for calling [crate::interpret::Machine::enter_trace_span] on a [tracing::info_span!].
117/// This is supposed to be compiled out when [crate::interpret::Machine::enter_trace_span] has the
118/// default implementation (i.e. when it does not actually enter the span but instead returns `()`).
119/// This macro takes a type implementing the [crate::interpret::Machine] trait as its first argument
120/// and otherwise accepts the same syntax as [tracing::span!] (see some tips below).
121/// Note: the result of this macro **must be used** because the span is exited when it's dropped.
122///
123/// ### Syntax accepted by this macro
124///
125/// The full documentation for the [tracing::span!] syntax can be found at [tracing] under "Using the
126/// Macros". A few possibly confusing syntaxes are listed here:
127/// ```rust
128/// # use rustc_const_eval::enter_trace_span;
129/// # type M = rustc_const_eval::const_eval::CompileTimeMachine<'static>;
130/// # let my_display_var = String::new();
131/// # let my_debug_var = String::new();
132/// // logs a span named "hello" with a field named "arg" of value 42 (works only because
133/// // 42 implements the tracing::Value trait, otherwise use one of the options below)
134/// let _trace = enter_trace_span!(M, "hello", arg = 42);
135/// // logs a field called "my_display_var" using the Display implementation
136/// let _trace = enter_trace_span!(M, "hello", %my_display_var);
137/// // logs a field called "my_debug_var" using the Debug implementation
138/// let _trace = enter_trace_span!(M, "hello", ?my_debug_var);
139/// ```
140///
141/// ### `NAME::SUBNAME` syntax
142///
143/// In addition to the syntax accepted by [tracing::span!], this macro optionally allows passing
144/// the span name (i.e. the first macro argument) in the form `NAME::SUBNAME` (without quotes) to
145/// indicate that the span has name "NAME" (usually the name of the component) and has an additional
146/// more specific name "SUBNAME" (usually the function name). The latter is passed to the [tracing]
147/// infrastructure as a span field with the name "NAME". This allows not being distracted by
148/// subnames when looking at the trace in <https://ui.perfetto.dev>, but when deeper introspection
149/// is needed within a component, it's still possible to view the subnames directly in the UI by
150/// selecting a span, clicking on the "NAME" argument on the right, and clicking on "Visualize
151/// argument values".
152/// ```rust
153/// # use rustc_const_eval::enter_trace_span;
154/// # type M = rustc_const_eval::const_eval::CompileTimeMachine<'static>;
155/// // for example, the first will expand to the second
156/// let _trace = enter_trace_span!(M, borrow_tracker::on_stack_pop, /* ... */);
157/// let _trace = enter_trace_span!(M, "borrow_tracker", borrow_tracker = "on_stack_pop", /* ... */);
158/// ```
159///
160/// ### `tracing_separate_thread` parameter
161///
162/// This macro was introduced to obtain better traces of Miri without impacting release performance.
163/// Miri saves traces using the `tracing_chrome` `tracing::Layer` so that they can be visualized
164/// in <https://ui.perfetto.dev>. To instruct `tracing_chrome` to put some spans on a separate trace
165/// thread/line than other spans when viewed in <https://ui.perfetto.dev>, you can pass
166/// `tracing_separate_thread = tracing::field::Empty` to the tracing macros. This is useful to
167/// separate out spans which just indicate the current step or program frame being processed by the
168/// interpreter. You should use a value of [tracing::field::Empty] so that other tracing layers
169/// (e.g. the logger) will ignore the `tracing_separate_thread` field. For example:
170/// ```rust
171/// # use rustc_const_eval::enter_trace_span;
172/// # type M = rustc_const_eval::const_eval::CompileTimeMachine<'static>;
173/// let _trace = enter_trace_span!(M, step::eval_statement, tracing_separate_thread = tracing::field::Empty);
174/// ```
175///
176/// ### Executing something else when tracing is disabled
177///
178/// [crate::interpret::Machine::enter_trace_span] returns [EnteredTraceSpan], on which you can call
179/// [EnteredTraceSpan::or_if_tracing_disabled], to e.g. log a line as an alternative to the tracing
180/// span for when tracing is disabled. For example:
181/// ```rust
182/// # use rustc_const_eval::enter_trace_span;
183/// # use rustc_const_eval::interpret::EnteredTraceSpan;
184/// # type M = rustc_const_eval::const_eval::CompileTimeMachine<'static>;
185/// let _trace = enter_trace_span!(M, step::eval_statement)
186/// .or_if_tracing_disabled(|| tracing::info!("eval_statement"));
187/// ```
188#[macro_export]
189macro_rules!enter_trace_span {
190 ($machine:ty, $name:ident :: $subname:ident $($tt:tt)*) => {
191$crate::enter_trace_span!($machine, stringify!($name), $name = %stringify!($subname) $($tt)*)
192 };
193194 ($machine:ty, $($tt:tt)*) => {
195 <$machine as $crate::interpret::Machine>::enter_trace_span(|| tracing::info_span!($($tt)*))
196 };
197}