Skip to content

Commit 488381c

Browse files
committed
Auto merge of #64419 - wesleywiser:const_prop_use_ecx, r=oli-obk
Deduplicate some code between miri and const prop r? @oli-obk
2 parents 18f00b9 + ba2d6c4 commit 488381c

27 files changed

+425
-254
lines changed

src/librustc/mir/interpret/error.rs

+6
Original file line numberDiff line numberDiff line change
@@ -389,6 +389,10 @@ pub enum UnsupportedOpInfo<'tcx> {
389389
/// Free-form case. Only for errors that are never caught!
390390
Unsupported(String),
391391

392+
/// FIXME(#64506) Error used to work around accessing projections of
393+
/// uninhabited types.
394+
UninhabitedValue,
395+
392396
// -- Everything below is not categorized yet --
393397
FunctionAbiMismatch(Abi, Abi),
394398
FunctionArgMismatch(Ty<'tcx>, Ty<'tcx>),
@@ -552,6 +556,8 @@ impl fmt::Debug for UnsupportedOpInfo<'tcx> {
552556
not a power of two"),
553557
Unsupported(ref msg) =>
554558
write!(f, "{}", msg),
559+
UninhabitedValue =>
560+
write!(f, "tried to use an uninhabited value"),
555561
}
556562
}
557563
}

src/librustc_mir/const_eval.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ use syntax::source_map::{Span, DUMMY_SP};
2121

2222
use crate::interpret::{self,
2323
PlaceTy, MPlaceTy, OpTy, ImmTy, Immediate, Scalar, Pointer,
24-
RawConst, ConstValue,
24+
RawConst, ConstValue, Machine,
2525
InterpResult, InterpErrorInfo, GlobalId, InterpCx, StackPopCleanup,
2626
Allocation, AllocId, MemoryKind, Memory,
2727
snapshot, RefTracking, intern_const_alloc_recursive,
@@ -41,7 +41,7 @@ const DETECTOR_SNAPSHOT_PERIOD: isize = 256;
4141
/// that inform us about the generic bounds of the constant. E.g., using an associated constant
4242
/// of a function's generic parameter will require knowledge about the bounds on the generic
4343
/// parameter. These bounds are passed to `mk_eval_cx` via the `ParamEnv` argument.
44-
pub(crate) fn mk_eval_cx<'mir, 'tcx>(
44+
fn mk_eval_cx<'mir, 'tcx>(
4545
tcx: TyCtxt<'tcx>,
4646
span: Span,
4747
param_env: ty::ParamEnv<'tcx>,
@@ -169,7 +169,7 @@ fn eval_body_using_ecx<'mir, 'tcx>(
169169
}
170170

171171
#[derive(Clone, Debug)]
172-
enum ConstEvalError {
172+
pub enum ConstEvalError {
173173
NeedsRfc(String),
174174
}
175175

@@ -521,8 +521,8 @@ pub fn const_variant_index<'tcx>(
521521
/// Turn an interpreter error into something to report to the user.
522522
/// As a side-effect, if RUSTC_CTFE_BACKTRACE is set, this prints the backtrace.
523523
/// Should be called only if the error is actually going to to be reported!
524-
pub fn error_to_const_error<'mir, 'tcx>(
525-
ecx: &InterpCx<'mir, 'tcx, CompileTimeInterpreter<'mir, 'tcx>>,
524+
pub fn error_to_const_error<'mir, 'tcx, M: Machine<'mir, 'tcx>>(
525+
ecx: &InterpCx<'mir, 'tcx, M>,
526526
mut error: InterpErrorInfo<'tcx>,
527527
) -> ConstEvalErr<'tcx> {
528528
error.print_backtrace();

src/librustc_mir/interpret/machine.rs

+17
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ use rustc::ty::{self, Ty, TyCtxt};
1212
use super::{
1313
Allocation, AllocId, InterpResult, Scalar, AllocationExtra,
1414
InterpCx, PlaceTy, OpTy, ImmTy, MemoryKind, Pointer, Memory,
15+
Frame, Operand,
1516
};
1617

1718
/// Whether this kind of memory is allowed to leak
@@ -184,6 +185,22 @@ pub trait Machine<'mir, 'tcx>: Sized {
184185
dest: PlaceTy<'tcx, Self::PointerTag>,
185186
) -> InterpResult<'tcx>;
186187

188+
/// Called to read the specified `local` from the `frame`.
189+
fn access_local(
190+
_ecx: &InterpCx<'mir, 'tcx, Self>,
191+
frame: &Frame<'mir, 'tcx, Self::PointerTag, Self::FrameExtra>,
192+
local: mir::Local,
193+
) -> InterpResult<'tcx, Operand<Self::PointerTag>> {
194+
frame.locals[local].access()
195+
}
196+
197+
/// Called before a `StaticKind::Static` value is accessed.
198+
fn before_access_static(
199+
_allocation: &Allocation,
200+
) -> InterpResult<'tcx> {
201+
Ok(())
202+
}
203+
187204
/// Called to initialize the "extra" state of an allocation and make the pointers
188205
/// it contains (in relocations) tagged. The way we construct allocations is
189206
/// to always first construct it without extra and then add the extra.

src/librustc_mir/interpret/memory.rs

+2
Original file line numberDiff line numberDiff line change
@@ -462,6 +462,8 @@ impl<'mir, 'tcx, M: Machine<'mir, 'tcx>> Memory<'mir, 'tcx, M> {
462462
// Make sure we use the ID of the resolved memory, not the lazy one!
463463
let id = raw_const.alloc_id;
464464
let allocation = tcx.alloc_map.lock().unwrap_memory(id);
465+
466+
M::before_access_static(allocation)?;
465467
Cow::Borrowed(allocation)
466468
}
467469
}

src/librustc_mir/interpret/operand.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -458,7 +458,7 @@ impl<'mir, 'tcx, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
458458
// Do not read from ZST, they might not be initialized
459459
Operand::Immediate(Scalar::zst().into())
460460
} else {
461-
frame.locals[local].access()?
461+
M::access_local(&self, frame, local)?
462462
};
463463
Ok(OpTy { op, layout })
464464
}
@@ -481,7 +481,7 @@ impl<'mir, 'tcx, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
481481

482482
// Evaluate a place with the goal of reading from it. This lets us sometimes
483483
// avoid allocations.
484-
pub(super) fn eval_place_to_op(
484+
pub fn eval_place_to_op(
485485
&self,
486486
place: &mir::Place<'tcx>,
487487
layout: Option<TyLayout<'tcx>>,

src/librustc_mir/interpret/place.rs

+5-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ use rustc::mir;
99
use rustc::mir::interpret::truncate;
1010
use rustc::ty::{self, Ty};
1111
use rustc::ty::layout::{
12-
self, Size, Align, LayoutOf, TyLayout, HasDataLayout, VariantIdx, PrimitiveExt
12+
self, Size, Abi, Align, LayoutOf, TyLayout, HasDataLayout, VariantIdx, PrimitiveExt
1313
};
1414
use rustc::ty::TypeFoldable;
1515

@@ -385,6 +385,10 @@ where
385385
stride * field
386386
}
387387
layout::FieldPlacement::Union(count) => {
388+
// FIXME(#64506) `UninhabitedValue` can be removed when this issue is resolved
389+
if base.layout.abi == Abi::Uninhabited {
390+
throw_unsup!(UninhabitedValue);
391+
}
388392
assert!(field < count as u64,
389393
"Tried to access field {} of union with {} fields", field, count);
390394
// Offset is always 0

src/librustc_mir/interpret/step.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -132,7 +132,7 @@ impl<'mir, 'tcx, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
132132
///
133133
/// There is no separate `eval_rvalue` function. Instead, the code for handling each rvalue
134134
/// type writes its results directly into the memory specified by the place.
135-
fn eval_rvalue_into_place(
135+
pub fn eval_rvalue_into_place(
136136
&mut self,
137137
rvalue: &mir::Rvalue<'tcx>,
138138
place: &mir::Place<'tcx>,

0 commit comments

Comments
 (0)