Skip to content

Commit 59337cd

Browse files
committed
Auto merge of #91342 - RalfJung:fn-abi, r=eddyb,oli-obk
CTFE eval_fn_call: use FnAbi to determine argument skipping and compatibility This makes use of the `FnAbi` type in CTFE/Miri, which `@eddyb` has been saying for years is what we should do.^^ `FnAbi` is used to - determine which arguments to skip (rather than the previous heuristic of skipping ZST arguments with the Rust ABI) - impose further restrictions on whether caller and callee are consistent in how a given argument is passed I was hoping it would also simplify the code, but that is not the case -- the previous type compatibility checks are still required (AFAIK), only the ZST skipping is gone and that took barely any code. We also need some hacks because `FnAbi` assumes a certain way of implementing `caller_location` (by passing extra arguments), but Miri can just read the caller location from the call stack so it doesn't need those arguments. (The fact that every backend has to separately implement support for these arguments seems suboptimal -- looks like this might have been better implemented on the MIR level.) To avoid having to implement those unnecessary arguments in Miri, we just compute *whether* the argument is present on the caller/callee side, but don't actually pass that argument around. I have no idea if this looks the way `@eddyb` thinks it should look... but it makes Miri's test suite pass. ;) One of rustc's tests fails unfortunately (`ui/const-generics/issues/issue-67739.rs`), some const generic code that is evaluated too early -- I think that should raise `TooGeneric` but instead it ICEs. My assumption is this is some FnAbi code that has not been properly tested on polymorphic code, but it might also be me calling that FnAbi code the wrong way. r? `@oli-obk` `@eddyb` Fixes #56166 Miri PR at rust-lang/miri#1928
2 parents e6f1f04 + 56b7d5f commit 59337cd

File tree

9 files changed

+238
-147
lines changed

9 files changed

+238
-147
lines changed

compiler/rustc_const_eval/src/const_eval/machine.rs

+13-3
Original file line numberDiff line numberDiff line change
@@ -260,7 +260,7 @@ impl<'mir, 'tcx> interpret::Machine<'mir, 'tcx> for CompileTimeInterpreter<'mir,
260260
args: &[OpTy<'tcx>],
261261
_ret: Option<(&PlaceTy<'tcx>, mir::BasicBlock)>,
262262
_unwind: StackPopUnwind, // unwinding is not supported in consts
263-
) -> InterpResult<'tcx, Option<&'mir mir::Body<'tcx>>> {
263+
) -> InterpResult<'tcx, Option<(&'mir mir::Body<'tcx>, ty::Instance<'tcx>)>> {
264264
debug!("find_mir_or_eval_fn: {:?}", instance);
265265

266266
// Only check non-glue functions
@@ -279,11 +279,21 @@ impl<'mir, 'tcx> interpret::Machine<'mir, 'tcx> for CompileTimeInterpreter<'mir,
279279

280280
if let Some(new_instance) = ecx.hook_special_const_fn(instance, args)? {
281281
// We call another const fn instead.
282-
return Self::find_mir_or_eval_fn(ecx, new_instance, _abi, args, _ret, _unwind);
282+
// However, we return the *original* instance to make backtraces work out
283+
// (and we hope this does not confuse the FnAbi checks too much).
284+
return Ok(Self::find_mir_or_eval_fn(
285+
ecx,
286+
new_instance,
287+
_abi,
288+
args,
289+
_ret,
290+
_unwind,
291+
)?
292+
.map(|(body, _instance)| (body, instance)));
283293
}
284294
}
285295
// This is a const fn. Call it.
286-
Ok(Some(ecx.load_mir(instance.def, None)?))
296+
Ok(Some((ecx.load_mir(instance.def, None)?, instance)))
287297
}
288298

289299
fn call_intrinsic(

compiler/rustc_const_eval/src/interpret/eval_context.rs

+23-2
Original file line numberDiff line numberDiff line change
@@ -8,15 +8,18 @@ use rustc_index::vec::IndexVec;
88
use rustc_macros::HashStable;
99
use rustc_middle::mir;
1010
use rustc_middle::mir::interpret::{InterpError, InvalidProgramInfo};
11-
use rustc_middle::ty::layout::{self, LayoutError, LayoutOf, LayoutOfHelpers, TyAndLayout};
11+
use rustc_middle::ty::layout::{
12+
self, FnAbiError, FnAbiOfHelpers, FnAbiRequest, LayoutError, LayoutOf, LayoutOfHelpers,
13+
TyAndLayout,
14+
};
1215
use rustc_middle::ty::{
1316
self, query::TyCtxtAt, subst::SubstsRef, ParamEnv, Ty, TyCtxt, TypeFoldable,
1417
};
1518
use rustc_mir_dataflow::storage::AlwaysLiveLocals;
1619
use rustc_query_system::ich::StableHashingContext;
1720
use rustc_session::Limit;
1821
use rustc_span::{Pos, Span};
19-
use rustc_target::abi::{Align, HasDataLayout, Size, TargetDataLayout};
22+
use rustc_target::abi::{call::FnAbi, Align, HasDataLayout, Size, TargetDataLayout};
2023

2124
use super::{
2225
AllocId, GlobalId, Immediate, InterpErrorInfo, InterpResult, MPlaceTy, Machine, MemPlace,
@@ -333,6 +336,24 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> LayoutOfHelpers<'tcx> for InterpC
333336
}
334337
}
335338

339+
impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> FnAbiOfHelpers<'tcx> for InterpCx<'mir, 'tcx, M> {
340+
type FnAbiOfResult = InterpResult<'tcx, &'tcx FnAbi<'tcx, Ty<'tcx>>>;
341+
342+
fn handle_fn_abi_err(
343+
&self,
344+
err: FnAbiError<'tcx>,
345+
_span: Span,
346+
_fn_abi_request: FnAbiRequest<'tcx>,
347+
) -> InterpErrorInfo<'tcx> {
348+
match err {
349+
FnAbiError::Layout(err) => err_inval!(Layout(err)).into(),
350+
FnAbiError::AdjustForForeignAbi(err) => {
351+
err_inval!(FnAbiAdjustForForeignAbi(err)).into()
352+
}
353+
}
354+
}
355+
}
356+
336357
/// Test if it is valid for a MIR assignment to assign `src`-typed place to `dest`-typed value.
337358
/// This test should be symmetric, as it is primarily about layout compatibility.
338359
pub(super) fn mir_assign_valid_types<'tcx>(

compiler/rustc_const_eval/src/interpret/machine.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -167,7 +167,7 @@ pub trait Machine<'mir, 'tcx>: Sized {
167167
args: &[OpTy<'tcx, Self::PointerTag>],
168168
ret: Option<(&PlaceTy<'tcx, Self::PointerTag>, mir::BasicBlock)>,
169169
unwind: StackPopUnwind,
170-
) -> InterpResult<'tcx, Option<&'mir mir::Body<'tcx>>>;
170+
) -> InterpResult<'tcx, Option<(&'mir mir::Body<'tcx>, ty::Instance<'tcx>)>>;
171171

172172
/// Execute `fn_val`. It is the hook's responsibility to advance the instruction
173173
/// pointer as appropriate.

0 commit comments

Comments
 (0)