mirror of
https://github.com/rust-lang/rust.git
synced 2024-12-11 08:05:12 +00:00
Auto merge of #55720 - RalfJung:const-eval-raw, r=oli-obk
Make const_eval_raw query return just an AllocId r? @oli-obk
This commit is contained in:
commit
c4cf115056
@ -317,6 +317,10 @@ impl_stable_hash_for!(
|
|||||||
ByRef(id, alloc, offset),
|
ByRef(id, alloc, offset),
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
impl_stable_hash_for!(struct ::mir::interpret::RawConst<'tcx> {
|
||||||
|
alloc_id,
|
||||||
|
ty,
|
||||||
|
});
|
||||||
|
|
||||||
impl_stable_hash_for! {
|
impl_stable_hash_for! {
|
||||||
impl<Tag> for struct mir::interpret::Pointer<Tag> {
|
impl<Tag> for struct mir::interpret::Pointer<Tag> {
|
||||||
|
@ -16,7 +16,7 @@ use ty::{self, Ty, layout};
|
|||||||
use ty::layout::{Size, Align, LayoutError};
|
use ty::layout::{Size, Align, LayoutError};
|
||||||
use rustc_target::spec::abi::Abi;
|
use rustc_target::spec::abi::Abi;
|
||||||
|
|
||||||
use super::{Pointer, InboundsCheck, ScalarMaybeUndef};
|
use super::{RawConst, Pointer, InboundsCheck, ScalarMaybeUndef};
|
||||||
|
|
||||||
use backtrace::Backtrace;
|
use backtrace::Backtrace;
|
||||||
|
|
||||||
@ -46,6 +46,7 @@ impl ErrorHandled {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub type ConstEvalRawResult<'tcx> = Result<RawConst<'tcx>, ErrorHandled>;
|
||||||
pub type ConstEvalResult<'tcx> = Result<&'tcx ty::Const<'tcx>, ErrorHandled>;
|
pub type ConstEvalResult<'tcx> = Result<&'tcx ty::Const<'tcx>, ErrorHandled>;
|
||||||
|
|
||||||
#[derive(Clone, Debug, RustcEncodable, RustcDecodable)]
|
#[derive(Clone, Debug, RustcEncodable, RustcDecodable)]
|
||||||
|
@ -22,10 +22,10 @@ mod pointer;
|
|||||||
|
|
||||||
pub use self::error::{
|
pub use self::error::{
|
||||||
EvalError, EvalResult, EvalErrorKind, AssertMessage, ConstEvalErr, struct_error,
|
EvalError, EvalResult, EvalErrorKind, AssertMessage, ConstEvalErr, struct_error,
|
||||||
FrameInfo, ConstEvalResult, ErrorHandled,
|
FrameInfo, ConstEvalRawResult, ConstEvalResult, ErrorHandled,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub use self::value::{Scalar, ConstValue, ScalarMaybeUndef};
|
pub use self::value::{Scalar, ScalarMaybeUndef, RawConst, ConstValue};
|
||||||
|
|
||||||
pub use self::allocation::{
|
pub use self::allocation::{
|
||||||
InboundsCheck, Allocation, AllocationExtra,
|
InboundsCheck, Allocation, AllocationExtra,
|
||||||
|
@ -10,12 +10,20 @@
|
|||||||
|
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
|
|
||||||
use ty::layout::{HasDataLayout, Size};
|
use crate::ty::{Ty, subst::Substs, layout::{HasDataLayout, Size}};
|
||||||
use ty::subst::Substs;
|
use crate::hir::def_id::DefId;
|
||||||
use hir::def_id::DefId;
|
|
||||||
|
|
||||||
use super::{EvalResult, Pointer, PointerArithmetic, Allocation, AllocId, sign_extend, truncate};
|
use super::{EvalResult, Pointer, PointerArithmetic, Allocation, AllocId, sign_extend, truncate};
|
||||||
|
|
||||||
|
/// Represents the result of a raw const operation, pre-validation.
|
||||||
|
#[derive(Copy, Clone, Debug, Eq, PartialEq, RustcEncodable, RustcDecodable, Hash)]
|
||||||
|
pub struct RawConst<'tcx> {
|
||||||
|
// the value lives here, at offset 0, and that allocation definitely is a `AllocType::Memory`
|
||||||
|
// (so you can use `AllocMap::unwrap_memory`).
|
||||||
|
pub alloc_id: AllocId,
|
||||||
|
pub ty: Ty<'tcx>,
|
||||||
|
}
|
||||||
|
|
||||||
/// Represents a constant value in Rust. Scalar and ScalarPair are optimizations which
|
/// Represents a constant value in Rust. Scalar and ScalarPair are optimizations which
|
||||||
/// matches the LocalValue optimizations for easy conversions between Value and ConstValue.
|
/// matches the LocalValue optimizations for easy conversions between Value and ConstValue.
|
||||||
#[derive(Copy, Clone, Debug, Eq, PartialEq, PartialOrd, Ord, RustcEncodable, RustcDecodable, Hash)]
|
#[derive(Copy, Clone, Debug, Eq, PartialEq, PartialOrd, Ord, RustcEncodable, RustcDecodable, Hash)]
|
||||||
@ -23,6 +31,7 @@ pub enum ConstValue<'tcx> {
|
|||||||
/// Never returned from the `const_eval` query, but the HIR contains these frequently in order
|
/// Never returned from the `const_eval` query, but the HIR contains these frequently in order
|
||||||
/// to allow HIR creation to happen for everything before needing to be able to run constant
|
/// to allow HIR creation to happen for everything before needing to be able to run constant
|
||||||
/// evaluation
|
/// evaluation
|
||||||
|
/// FIXME: The query should then return a type that does not even have this variant.
|
||||||
Unevaluated(DefId, &'tcx Substs<'tcx>),
|
Unevaluated(DefId, &'tcx Substs<'tcx>),
|
||||||
|
|
||||||
/// Used only for types with layout::abi::Scalar ABI and ZSTs
|
/// Used only for types with layout::abi::Scalar ABI and ZSTs
|
||||||
|
@ -27,7 +27,7 @@ use middle::stability::{self, DeprecationEntry};
|
|||||||
use middle::lib_features::LibFeatures;
|
use middle::lib_features::LibFeatures;
|
||||||
use middle::lang_items::{LanguageItems, LangItem};
|
use middle::lang_items::{LanguageItems, LangItem};
|
||||||
use middle::exported_symbols::{SymbolExportLevel, ExportedSymbol};
|
use middle::exported_symbols::{SymbolExportLevel, ExportedSymbol};
|
||||||
use mir::interpret::ConstEvalResult;
|
use mir::interpret::{ConstEvalRawResult, ConstEvalResult};
|
||||||
use mir::mono::CodegenUnit;
|
use mir::mono::CodegenUnit;
|
||||||
use mir;
|
use mir;
|
||||||
use mir::interpret::GlobalId;
|
use mir::interpret::GlobalId;
|
||||||
@ -309,7 +309,7 @@ define_queries! { <'tcx>
|
|||||||
/// validation. Please add a comment to every use site explaining why using `const_eval`
|
/// validation. Please add a comment to every use site explaining why using `const_eval`
|
||||||
/// isn't sufficient
|
/// isn't sufficient
|
||||||
[] fn const_eval_raw: const_eval_raw_dep_node(ty::ParamEnvAnd<'tcx, GlobalId<'tcx>>)
|
[] fn const_eval_raw: const_eval_raw_dep_node(ty::ParamEnvAnd<'tcx, GlobalId<'tcx>>)
|
||||||
-> ConstEvalResult<'tcx>,
|
-> ConstEvalRawResult<'tcx>,
|
||||||
|
|
||||||
/// Results of evaluating const items or constants embedded in
|
/// Results of evaluating const items or constants embedded in
|
||||||
/// other items (such as enum variant explicit discriminants).
|
/// other items (such as enum variant explicit discriminants).
|
||||||
|
@ -31,8 +31,8 @@ use rustc::util::common::ErrorReported;
|
|||||||
use syntax::ast::Mutability;
|
use syntax::ast::Mutability;
|
||||||
use syntax::source_map::{Span, DUMMY_SP};
|
use syntax::source_map::{Span, DUMMY_SP};
|
||||||
|
|
||||||
use interpret::{self,
|
use crate::interpret::{self,
|
||||||
PlaceTy, MemPlace, OpTy, Operand, Immediate, Scalar, ConstValue, Pointer,
|
PlaceTy, MPlaceTy, MemPlace, OpTy, Operand, Immediate, Scalar, RawConst, ConstValue, Pointer,
|
||||||
EvalResult, EvalError, EvalErrorKind, GlobalId, EvalContext, StackPopCleanup,
|
EvalResult, EvalError, EvalErrorKind, GlobalId, EvalContext, StackPopCleanup,
|
||||||
Allocation, AllocId, MemoryKind,
|
Allocation, AllocId, MemoryKind,
|
||||||
snapshot, RefTracking,
|
snapshot, RefTracking,
|
||||||
@ -94,11 +94,12 @@ pub(crate) fn eval_promoted<'a, 'mir, 'tcx>(
|
|||||||
cid: GlobalId<'tcx>,
|
cid: GlobalId<'tcx>,
|
||||||
mir: &'mir mir::Mir<'tcx>,
|
mir: &'mir mir::Mir<'tcx>,
|
||||||
param_env: ty::ParamEnv<'tcx>,
|
param_env: ty::ParamEnv<'tcx>,
|
||||||
) -> EvalResult<'tcx, OpTy<'tcx>> {
|
) -> EvalResult<'tcx, MPlaceTy<'tcx>> {
|
||||||
let mut ecx = mk_borrowck_eval_cx(tcx, cid.instance, mir, DUMMY_SP).unwrap();
|
let mut ecx = mk_borrowck_eval_cx(tcx, cid.instance, mir, DUMMY_SP).unwrap();
|
||||||
eval_body_using_ecx(&mut ecx, cid, Some(mir), param_env)
|
eval_body_using_ecx(&mut ecx, cid, Some(mir), param_env)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FIXME: These two conversion functions are bad hacks. We should just always use allocations.
|
||||||
pub fn op_to_const<'tcx>(
|
pub fn op_to_const<'tcx>(
|
||||||
ecx: &CompileTimeEvalContext<'_, '_, 'tcx>,
|
ecx: &CompileTimeEvalContext<'_, '_, 'tcx>,
|
||||||
op: OpTy<'tcx>,
|
op: OpTy<'tcx>,
|
||||||
@ -144,13 +145,20 @@ pub fn op_to_const<'tcx>(
|
|||||||
};
|
};
|
||||||
Ok(ty::Const::from_const_value(ecx.tcx.tcx, val, op.layout.ty))
|
Ok(ty::Const::from_const_value(ecx.tcx.tcx, val, op.layout.ty))
|
||||||
}
|
}
|
||||||
|
pub fn const_to_op<'tcx>(
|
||||||
|
ecx: &CompileTimeEvalContext<'_, '_, 'tcx>,
|
||||||
|
cnst: &ty::Const<'tcx>,
|
||||||
|
) -> EvalResult<'tcx, OpTy<'tcx>> {
|
||||||
|
let op = ecx.const_value_to_op(cnst.val)?;
|
||||||
|
Ok(OpTy { op, layout: ecx.layout_of(cnst.ty)? })
|
||||||
|
}
|
||||||
|
|
||||||
fn eval_body_and_ecx<'a, 'mir, 'tcx>(
|
fn eval_body_and_ecx<'a, 'mir, 'tcx>(
|
||||||
tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
||||||
cid: GlobalId<'tcx>,
|
cid: GlobalId<'tcx>,
|
||||||
mir: Option<&'mir mir::Mir<'tcx>>,
|
mir: Option<&'mir mir::Mir<'tcx>>,
|
||||||
param_env: ty::ParamEnv<'tcx>,
|
param_env: ty::ParamEnv<'tcx>,
|
||||||
) -> (EvalResult<'tcx, OpTy<'tcx>>, CompileTimeEvalContext<'a, 'mir, 'tcx>) {
|
) -> (EvalResult<'tcx, MPlaceTy<'tcx>>, CompileTimeEvalContext<'a, 'mir, 'tcx>) {
|
||||||
// we start out with the best span we have
|
// we start out with the best span we have
|
||||||
// and try improving it down the road when more information is available
|
// and try improving it down the road when more information is available
|
||||||
let span = tcx.def_span(cid.instance.def_id());
|
let span = tcx.def_span(cid.instance.def_id());
|
||||||
@ -166,7 +174,7 @@ fn eval_body_using_ecx<'mir, 'tcx>(
|
|||||||
cid: GlobalId<'tcx>,
|
cid: GlobalId<'tcx>,
|
||||||
mir: Option<&'mir mir::Mir<'tcx>>,
|
mir: Option<&'mir mir::Mir<'tcx>>,
|
||||||
param_env: ty::ParamEnv<'tcx>,
|
param_env: ty::ParamEnv<'tcx>,
|
||||||
) -> EvalResult<'tcx, OpTy<'tcx>> {
|
) -> EvalResult<'tcx, MPlaceTy<'tcx>> {
|
||||||
debug!("eval_body_using_ecx: {:?}, {:?}", cid, param_env);
|
debug!("eval_body_using_ecx: {:?}, {:?}", cid, param_env);
|
||||||
let tcx = ecx.tcx.tcx;
|
let tcx = ecx.tcx.tcx;
|
||||||
let mut mir = match mir {
|
let mut mir = match mir {
|
||||||
@ -206,7 +214,7 @@ fn eval_body_using_ecx<'mir, 'tcx>(
|
|||||||
ecx.memory.intern_static(ret.ptr.to_ptr()?.alloc_id, mutability)?;
|
ecx.memory.intern_static(ret.ptr.to_ptr()?.alloc_id, mutability)?;
|
||||||
|
|
||||||
debug!("eval_body_using_ecx done: {:?}", *ret);
|
debug!("eval_body_using_ecx done: {:?}", *ret);
|
||||||
Ok(ret.into())
|
Ok(ret)
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'tcx> Into<EvalError<'tcx>> for ConstEvalError {
|
impl<'tcx> Into<EvalError<'tcx>> for ConstEvalError {
|
||||||
@ -494,7 +502,7 @@ pub fn const_field<'a, 'tcx>(
|
|||||||
let ecx = mk_eval_cx(tcx, instance, param_env).unwrap();
|
let ecx = mk_eval_cx(tcx, instance, param_env).unwrap();
|
||||||
let result = (|| {
|
let result = (|| {
|
||||||
// get the operand again
|
// get the operand again
|
||||||
let op = ecx.const_to_op(value)?;
|
let op = const_to_op(&ecx, value)?;
|
||||||
// downcast
|
// downcast
|
||||||
let down = match variant {
|
let down = match variant {
|
||||||
None => op,
|
None => op,
|
||||||
@ -521,7 +529,7 @@ pub fn const_variant_index<'a, 'tcx>(
|
|||||||
) -> EvalResult<'tcx, VariantIdx> {
|
) -> EvalResult<'tcx, VariantIdx> {
|
||||||
trace!("const_variant_index: {:?}, {:?}", instance, val);
|
trace!("const_variant_index: {:?}, {:?}", instance, val);
|
||||||
let ecx = mk_eval_cx(tcx, instance, param_env).unwrap();
|
let ecx = mk_eval_cx(tcx, instance, param_env).unwrap();
|
||||||
let op = ecx.const_to_op(val)?;
|
let op = const_to_op(&ecx, val)?;
|
||||||
Ok(ecx.read_discriminant(op)?.1)
|
Ok(ecx.read_discriminant(op)?.1)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -534,15 +542,17 @@ pub fn error_to_const_error<'a, 'mir, 'tcx>(
|
|||||||
ConstEvalErr { error: error.kind, stacktrace, span: ecx.tcx.span }
|
ConstEvalErr { error: error.kind, stacktrace, span: ecx.tcx.span }
|
||||||
}
|
}
|
||||||
|
|
||||||
fn validate_const<'a, 'tcx>(
|
fn validate_and_turn_into_const<'a, 'tcx>(
|
||||||
tcx: ty::TyCtxt<'a, 'tcx, 'tcx>,
|
tcx: ty::TyCtxt<'a, 'tcx, 'tcx>,
|
||||||
constant: &'tcx ty::Const<'tcx>,
|
constant: RawConst<'tcx>,
|
||||||
key: ty::ParamEnvAnd<'tcx, GlobalId<'tcx>>,
|
key: ty::ParamEnvAnd<'tcx, GlobalId<'tcx>>,
|
||||||
) -> ::rustc::mir::interpret::ConstEvalResult<'tcx> {
|
) -> ::rustc::mir::interpret::ConstEvalResult<'tcx> {
|
||||||
let cid = key.value;
|
let cid = key.value;
|
||||||
let ecx = mk_eval_cx(tcx, cid.instance, key.param_env).unwrap();
|
let ecx = mk_eval_cx(tcx, cid.instance, key.param_env).unwrap();
|
||||||
let val = (|| {
|
let val = (|| {
|
||||||
let op = ecx.const_to_op(constant)?;
|
let op = ecx.raw_const_to_mplace(constant)?.into();
|
||||||
|
// FIXME: Once the visitor infrastructure landed, change validation to
|
||||||
|
// work directly on `MPlaceTy`.
|
||||||
let mut ref_tracking = RefTracking::new(op);
|
let mut ref_tracking = RefTracking::new(op);
|
||||||
while let Some((op, path)) = ref_tracking.todo.pop() {
|
while let Some((op, path)) = ref_tracking.todo.pop() {
|
||||||
ecx.validate_operand(
|
ecx.validate_operand(
|
||||||
@ -552,7 +562,10 @@ fn validate_const<'a, 'tcx>(
|
|||||||
/* const_mode */ true,
|
/* const_mode */ true,
|
||||||
)?;
|
)?;
|
||||||
}
|
}
|
||||||
Ok(constant)
|
// Now that we validated, turn this into a proper constant
|
||||||
|
let def_id = cid.instance.def.def_id();
|
||||||
|
let normalize = tcx.is_static(def_id).is_none() && cid.promoted.is_none();
|
||||||
|
op_to_const(&ecx, op, normalize)
|
||||||
})();
|
})();
|
||||||
|
|
||||||
val.map_err(|error| {
|
val.map_err(|error| {
|
||||||
@ -591,14 +604,14 @@ pub fn const_eval_provider<'a, 'tcx>(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
tcx.const_eval_raw(key).and_then(|val| {
|
tcx.const_eval_raw(key).and_then(|val| {
|
||||||
validate_const(tcx, val, key)
|
validate_and_turn_into_const(tcx, val, key)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn const_eval_raw_provider<'a, 'tcx>(
|
pub fn const_eval_raw_provider<'a, 'tcx>(
|
||||||
tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
||||||
key: ty::ParamEnvAnd<'tcx, GlobalId<'tcx>>,
|
key: ty::ParamEnvAnd<'tcx, GlobalId<'tcx>>,
|
||||||
) -> ::rustc::mir::interpret::ConstEvalResult<'tcx> {
|
) -> ::rustc::mir::interpret::ConstEvalRawResult<'tcx> {
|
||||||
// Because the constant is computed twice (once per value of `Reveal`), we are at risk of
|
// Because the constant is computed twice (once per value of `Reveal`), we are at risk of
|
||||||
// reporting the same error twice here. To resolve this, we check whether we can evaluate the
|
// reporting the same error twice here. To resolve this, we check whether we can evaluate the
|
||||||
// constant in the more restrictive `Reveal::UserFacing`, which most likely already was
|
// constant in the more restrictive `Reveal::UserFacing`, which most likely already was
|
||||||
@ -648,16 +661,11 @@ pub fn const_eval_raw_provider<'a, 'tcx>(
|
|||||||
};
|
};
|
||||||
|
|
||||||
let (res, ecx) = eval_body_and_ecx(tcx, cid, None, key.param_env);
|
let (res, ecx) = eval_body_and_ecx(tcx, cid, None, key.param_env);
|
||||||
res.and_then(|op| {
|
res.and_then(|place| {
|
||||||
let normalize = tcx.is_static(def_id).is_none() && cid.promoted.is_none();
|
Ok(RawConst {
|
||||||
if !normalize {
|
alloc_id: place.to_ptr().expect("we allocated this ptr!").alloc_id,
|
||||||
// Sanity check: These must always be a MemPlace
|
ty: place.layout.ty
|
||||||
match op.op {
|
})
|
||||||
Operand::Indirect(_) => { /* all is good */ },
|
|
||||||
Operand::Immediate(_) => bug!("const eval gave us an Immediate"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
op_to_const(&ecx, op, normalize)
|
|
||||||
}).map_err(|error| {
|
}).map_err(|error| {
|
||||||
let err = error_to_const_error(&ecx, error);
|
let err = error_to_const_error(&ecx, error);
|
||||||
// errors in statics are always emitted as fatal errors
|
// errors in statics are always emitted as fatal errors
|
||||||
|
@ -588,18 +588,26 @@ impl<'a, 'mir, 'tcx: 'mir, M: Machine<'a, 'mir, 'tcx>> EvalContext<'a, 'mir, 'tc
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn const_eval(&self, gid: GlobalId<'tcx>) -> EvalResult<'tcx, &'tcx ty::Const<'tcx>> {
|
pub fn const_eval_raw(
|
||||||
|
&self,
|
||||||
|
gid: GlobalId<'tcx>,
|
||||||
|
) -> EvalResult<'tcx, MPlaceTy<'tcx, M::PointerTag>> {
|
||||||
let param_env = if self.tcx.is_static(gid.instance.def_id()).is_some() {
|
let param_env = if self.tcx.is_static(gid.instance.def_id()).is_some() {
|
||||||
ty::ParamEnv::reveal_all()
|
ty::ParamEnv::reveal_all()
|
||||||
} else {
|
} else {
|
||||||
self.param_env
|
self.param_env
|
||||||
};
|
};
|
||||||
self.tcx.const_eval(param_env.and(gid)).map_err(|err| {
|
// We use `const_eval_raw` here, and get an unvalidated result. That is okay:
|
||||||
|
// Our result will later be validated anyway, and there seems no good reason
|
||||||
|
// to have to fail early here. This is also more consistent with
|
||||||
|
// `Memory::get_static_alloc` which has to use `const_eval_raw` to avoid cycles.
|
||||||
|
let val = self.tcx.const_eval_raw(param_env.and(gid)).map_err(|err| {
|
||||||
match err {
|
match err {
|
||||||
ErrorHandled::Reported => EvalErrorKind::ReferencedConstant.into(),
|
ErrorHandled::Reported => EvalErrorKind::ReferencedConstant,
|
||||||
ErrorHandled::TooGeneric => EvalErrorKind::TooGeneric.into(),
|
ErrorHandled::TooGeneric => EvalErrorKind::TooGeneric,
|
||||||
}
|
}
|
||||||
})
|
})?;
|
||||||
|
self.raw_const_to_mplace(val)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn dump_place(&self, place: Place<M::PointerTag>) {
|
pub fn dump_place(&self, place: Place<M::PointerTag>) {
|
||||||
|
@ -28,7 +28,7 @@ use rustc_data_structures::fx::{FxHashSet, FxHashMap};
|
|||||||
use syntax::ast::Mutability;
|
use syntax::ast::Mutability;
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
Pointer, AllocId, Allocation, ConstValue, GlobalId, AllocationExtra, InboundsCheck,
|
Pointer, AllocId, Allocation, GlobalId, AllocationExtra, InboundsCheck,
|
||||||
EvalResult, Scalar, EvalErrorKind, AllocType, PointerArithmetic,
|
EvalResult, Scalar, EvalErrorKind, AllocType, PointerArithmetic,
|
||||||
Machine, AllocMap, MayLeak, ScalarMaybeUndef, ErrorHandled,
|
Machine, AllocMap, MayLeak, ScalarMaybeUndef, ErrorHandled,
|
||||||
};
|
};
|
||||||
@ -374,14 +374,11 @@ impl<'a, 'mir, 'tcx, M: Machine<'a, 'mir, 'tcx>> Memory<'a, 'mir, 'tcx, M> {
|
|||||||
ErrorHandled::Reported => EvalErrorKind::ReferencedConstant.into(),
|
ErrorHandled::Reported => EvalErrorKind::ReferencedConstant.into(),
|
||||||
ErrorHandled::TooGeneric => EvalErrorKind::TooGeneric.into(),
|
ErrorHandled::TooGeneric => EvalErrorKind::TooGeneric.into(),
|
||||||
}
|
}
|
||||||
}).map(|const_val| {
|
}).map(|raw_const| {
|
||||||
if let ConstValue::ByRef(_, allocation, _) = const_val.val {
|
let allocation = tcx.alloc_map.lock().unwrap_memory(raw_const.alloc_id);
|
||||||
// We got tcx memory. Let the machine figure out whether and how to
|
// We got tcx memory. Let the machine figure out whether and how to
|
||||||
// turn that into memory with the right pointer tag.
|
// turn that into memory with the right pointer tag.
|
||||||
M::adjust_static_allocation(allocation)
|
M::adjust_static_allocation(allocation)
|
||||||
} else {
|
|
||||||
bug!("Matching on non-ByRef static")
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,7 +13,7 @@
|
|||||||
|
|
||||||
use std::convert::TryInto;
|
use std::convert::TryInto;
|
||||||
|
|
||||||
use rustc::{mir, ty};
|
use rustc::mir;
|
||||||
use rustc::ty::layout::{self, Size, LayoutOf, TyLayout, HasDataLayout, IntegerExt, VariantIdx};
|
use rustc::ty::layout::{self, Size, LayoutOf, TyLayout, HasDataLayout, IntegerExt, VariantIdx};
|
||||||
|
|
||||||
use rustc::mir::interpret::{
|
use rustc::mir::interpret::{
|
||||||
@ -535,8 +535,10 @@ impl<'a, 'mir, 'tcx, M: Machine<'a, 'mir, 'tcx>> EvalContext<'a, 'mir, 'tcx, M>
|
|||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Also used e.g. when miri runs into a constant.
|
// Used when miri runs into a constant, and by CTFE.
|
||||||
pub(super) fn const_value_to_op(
|
// FIXME: CTFE should use allocations, then we can make this private (embed it into
|
||||||
|
// `eval_operand`, ideally).
|
||||||
|
pub(crate) fn const_value_to_op(
|
||||||
&self,
|
&self,
|
||||||
val: ConstValue<'tcx>,
|
val: ConstValue<'tcx>,
|
||||||
) -> EvalResult<'tcx, Operand<M::PointerTag>> {
|
) -> EvalResult<'tcx, Operand<M::PointerTag>> {
|
||||||
@ -544,10 +546,10 @@ impl<'a, 'mir, 'tcx, M: Machine<'a, 'mir, 'tcx>> EvalContext<'a, 'mir, 'tcx, M>
|
|||||||
match val {
|
match val {
|
||||||
ConstValue::Unevaluated(def_id, substs) => {
|
ConstValue::Unevaluated(def_id, substs) => {
|
||||||
let instance = self.resolve(def_id, substs)?;
|
let instance = self.resolve(def_id, substs)?;
|
||||||
self.global_to_op(GlobalId {
|
Ok(*OpTy::from(self.const_eval_raw(GlobalId {
|
||||||
instance,
|
instance,
|
||||||
promoted: None,
|
promoted: None,
|
||||||
})
|
})?))
|
||||||
}
|
}
|
||||||
ConstValue::ByRef(id, alloc, offset) => {
|
ConstValue::ByRef(id, alloc, offset) => {
|
||||||
// We rely on mutability being set correctly in that allocation to prevent writes
|
// We rely on mutability being set correctly in that allocation to prevent writes
|
||||||
@ -565,21 +567,6 @@ impl<'a, 'mir, 'tcx, M: Machine<'a, 'mir, 'tcx>> EvalContext<'a, 'mir, 'tcx, M>
|
|||||||
Ok(Operand::Immediate(Immediate::Scalar(x.into())).with_default_tag()),
|
Ok(Operand::Immediate(Immediate::Scalar(x.into())).with_default_tag()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pub fn const_to_op(
|
|
||||||
&self,
|
|
||||||
cnst: &ty::Const<'tcx>,
|
|
||||||
) -> EvalResult<'tcx, OpTy<'tcx, M::PointerTag>> {
|
|
||||||
let op = self.const_value_to_op(cnst.val)?;
|
|
||||||
Ok(OpTy { op, layout: self.layout_of(cnst.ty)? })
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(super) fn global_to_op(
|
|
||||||
&self,
|
|
||||||
gid: GlobalId<'tcx>
|
|
||||||
) -> EvalResult<'tcx, Operand<M::PointerTag>> {
|
|
||||||
let cv = self.const_eval(gid)?;
|
|
||||||
self.const_value_to_op(cv.val)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Read discriminant, return the runtime value as well as the variant index.
|
/// Read discriminant, return the runtime value as well as the variant index.
|
||||||
pub fn read_discriminant(
|
pub fn read_discriminant(
|
||||||
|
@ -20,12 +20,10 @@ use rustc::mir;
|
|||||||
use rustc::ty::{self, Ty};
|
use rustc::ty::{self, Ty};
|
||||||
use rustc::ty::layout::{self, Size, Align, LayoutOf, TyLayout, HasDataLayout, VariantIdx};
|
use rustc::ty::layout::{self, Size, Align, LayoutOf, TyLayout, HasDataLayout, VariantIdx};
|
||||||
|
|
||||||
use rustc::mir::interpret::{
|
|
||||||
GlobalId, AllocId, Allocation, Scalar, EvalResult, Pointer, PointerArithmetic
|
|
||||||
};
|
|
||||||
use super::{
|
use super::{
|
||||||
|
GlobalId, AllocId, Allocation, Scalar, EvalResult, Pointer, PointerArithmetic,
|
||||||
EvalContext, Machine, AllocMap, AllocationExtra,
|
EvalContext, Machine, AllocMap, AllocationExtra,
|
||||||
Immediate, ImmTy, ScalarMaybeUndef, Operand, OpTy, MemoryKind
|
RawConst, Immediate, ImmTy, ScalarMaybeUndef, Operand, OpTy, MemoryKind
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, Hash, PartialEq, Eq)]
|
#[derive(Copy, Clone, Debug, Hash, PartialEq, Eq)]
|
||||||
@ -555,16 +553,10 @@ where
|
|||||||
Ok(match *mir_place {
|
Ok(match *mir_place {
|
||||||
Promoted(ref promoted) => {
|
Promoted(ref promoted) => {
|
||||||
let instance = self.frame().instance;
|
let instance = self.frame().instance;
|
||||||
let op = self.global_to_op(GlobalId {
|
self.const_eval_raw(GlobalId {
|
||||||
instance,
|
instance,
|
||||||
promoted: Some(promoted.0),
|
promoted: Some(promoted.0),
|
||||||
})?;
|
})?
|
||||||
let mplace = op.to_mem_place(); // these are always in memory
|
|
||||||
let ty = self.monomorphize(promoted.1, self.substs());
|
|
||||||
MPlaceTy {
|
|
||||||
mplace,
|
|
||||||
layout: self.layout_of(ty)?,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Static(ref static_) => {
|
Static(ref static_) => {
|
||||||
@ -981,6 +973,19 @@ where
|
|||||||
Ok(OpTy { op, layout: place.layout })
|
Ok(OpTy { op, layout: place.layout })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn raw_const_to_mplace(
|
||||||
|
&self,
|
||||||
|
raw: RawConst<'tcx>,
|
||||||
|
) -> EvalResult<'tcx, MPlaceTy<'tcx, M::PointerTag>> {
|
||||||
|
// This must be an allocation in `tcx`
|
||||||
|
assert!(self.tcx.alloc_map.lock().get(raw.alloc_id).is_some());
|
||||||
|
let layout = self.layout_of(raw.ty)?;
|
||||||
|
Ok(MPlaceTy::from_aligned_ptr(
|
||||||
|
Pointer::new(raw.alloc_id, Size::ZERO).with_default_tag(),
|
||||||
|
layout,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
/// Turn a place with a `dyn Trait` type into a place with the actual dynamic type.
|
/// Turn a place with a `dyn Trait` type into a place with the actual dynamic type.
|
||||||
/// Also return some more information so drop doesn't have to run the same code twice.
|
/// Also return some more information so drop doesn't have to run the same code twice.
|
||||||
pub(super) fn unpack_dyn_trait(&self, mplace: MPlaceTy<'tcx, M::PointerTag>)
|
pub(super) fn unpack_dyn_trait(&self, mplace: MPlaceTy<'tcx, M::PointerTag>)
|
||||||
|
@ -29,7 +29,9 @@ use rustc::ty::layout::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use interpret::{self, EvalContext, ScalarMaybeUndef, Immediate, OpTy, MemoryKind};
|
use interpret::{self, EvalContext, ScalarMaybeUndef, Immediate, OpTy, MemoryKind};
|
||||||
use const_eval::{CompileTimeInterpreter, error_to_const_error, eval_promoted, mk_borrowck_eval_cx};
|
use const_eval::{
|
||||||
|
CompileTimeInterpreter, const_to_op, error_to_const_error, eval_promoted, mk_borrowck_eval_cx
|
||||||
|
};
|
||||||
use transform::{MirPass, MirSource};
|
use transform::{MirPass, MirSource};
|
||||||
|
|
||||||
pub struct ConstProp;
|
pub struct ConstProp;
|
||||||
@ -262,7 +264,7 @@ impl<'a, 'mir, 'tcx> ConstPropagator<'a, 'mir, 'tcx> {
|
|||||||
source_info: SourceInfo,
|
source_info: SourceInfo,
|
||||||
) -> Option<Const<'tcx>> {
|
) -> Option<Const<'tcx>> {
|
||||||
self.ecx.tcx.span = source_info.span;
|
self.ecx.tcx.span = source_info.span;
|
||||||
match self.ecx.const_to_op(c.literal) {
|
match const_to_op(&self.ecx, c.literal) {
|
||||||
Ok(op) => {
|
Ok(op) => {
|
||||||
Some((op, c.span))
|
Some((op, c.span))
|
||||||
},
|
},
|
||||||
@ -309,7 +311,7 @@ impl<'a, 'mir, 'tcx> ConstPropagator<'a, 'mir, 'tcx> {
|
|||||||
eval_promoted(this.tcx, cid, this.mir, this.param_env)
|
eval_promoted(this.tcx, cid, this.mir, this.param_env)
|
||||||
})?;
|
})?;
|
||||||
trace!("evaluated promoted {:?} to {:?}", promoted, res);
|
trace!("evaluated promoted {:?} to {:?}", promoted, res);
|
||||||
Some((res, source_info.span))
|
Some((res.into(), source_info.span))
|
||||||
},
|
},
|
||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,7 @@ union Foo {
|
|||||||
|
|
||||||
enum Bar {
|
enum Bar {
|
||||||
Boo = [unsafe { Foo { b: () }.a }; 4][3],
|
Boo = [unsafe { Foo { b: () }.a }; 4][3],
|
||||||
//~^ ERROR evaluation of constant value failed
|
//~^ ERROR it is undefined behavior to use this value
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
error[E0080]: evaluation of constant value failed
|
error[E0080]: it is undefined behavior to use this value
|
||||||
--> $DIR/const-err4.rs:18:11
|
--> $DIR/const-err4.rs:18:11
|
||||||
|
|
|
|
||||||
LL | Boo = [unsafe { Foo { b: () }.a }; 4][3],
|
LL | Boo = [unsafe { Foo { b: () }.a }; 4][3],
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ attempted to read undefined bytes
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type validation failed: encountered uninitialized bytes, but expected initialized plain (non-pointer) bytes
|
||||||
|
|
|
||||||
|
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
||||||
|
|
||||||
error: aborting due to previous error
|
error: aborting due to previous error
|
||||||
|
|
||||||
|
@ -37,7 +37,7 @@ fn main() {
|
|||||||
//~^ ERROR it is undefined behavior to use this value
|
//~^ ERROR it is undefined behavior to use this value
|
||||||
|
|
||||||
const I32_REF_U128_UNION: u128 = unsafe { Nonsense { int_32_ref: &3 }.uint_128 };
|
const I32_REF_U128_UNION: u128 = unsafe { Nonsense { int_32_ref: &3 }.uint_128 };
|
||||||
//~^ ERROR any use of this value will cause an error
|
//~^ ERROR it is undefined behavior to use this value
|
||||||
|
|
||||||
const I32_REF_I8_UNION: i8 = unsafe { Nonsense { int_32_ref: &3 }.int_8 };
|
const I32_REF_I8_UNION: i8 = unsafe { Nonsense { int_32_ref: &3 }.int_8 };
|
||||||
//~^ ERROR any use of this value will cause an error
|
//~^ ERROR any use of this value will cause an error
|
||||||
@ -52,7 +52,7 @@ fn main() {
|
|||||||
//~^ ERROR it is undefined behavior to use this value
|
//~^ ERROR it is undefined behavior to use this value
|
||||||
|
|
||||||
const I32_REF_I128_UNION: i128 = unsafe { Nonsense { int_32_ref: &3 }.int_128 };
|
const I32_REF_I128_UNION: i128 = unsafe { Nonsense { int_32_ref: &3 }.int_128 };
|
||||||
//~^ ERROR any use of this value will cause an error
|
//~^ ERROR it is undefined behavior to use this value
|
||||||
|
|
||||||
const I32_REF_F32_UNION: f32 = unsafe { Nonsense { int_32_ref: &3 }.float_32 };
|
const I32_REF_F32_UNION: f32 = unsafe { Nonsense { int_32_ref: &3 }.float_32 };
|
||||||
//~^ ERROR any use of this value will cause an error
|
//~^ ERROR any use of this value will cause an error
|
||||||
|
@ -40,11 +40,13 @@ LL | const I32_REF_U64_UNION: u64 = unsafe { Nonsense { int_32_ref: &3 }.uin
|
|||||||
|
|
|
|
||||||
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
||||||
|
|
||||||
error: any use of this value will cause an error
|
error[E0080]: it is undefined behavior to use this value
|
||||||
--> $DIR/const-pointer-values-in-various-types.rs:39:5
|
--> $DIR/const-pointer-values-in-various-types.rs:39:5
|
||||||
|
|
|
|
||||||
LL | const I32_REF_U128_UNION: u128 = unsafe { Nonsense { int_32_ref: &3 }.uint_128 };
|
LL | const I32_REF_U128_UNION: u128 = unsafe { Nonsense { int_32_ref: &3 }.uint_128 };
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ attempted to read undefined bytes
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type validation failed: encountered uninitialized bytes, but expected initialized plain (non-pointer) bytes
|
||||||
|
|
|
||||||
|
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
||||||
|
|
||||||
error: any use of this value will cause an error
|
error: any use of this value will cause an error
|
||||||
--> $DIR/const-pointer-values-in-various-types.rs:42:5
|
--> $DIR/const-pointer-values-in-various-types.rs:42:5
|
||||||
@ -78,11 +80,13 @@ LL | const I32_REF_I64_UNION: i64 = unsafe { Nonsense { int_32_ref: &3 }.int
|
|||||||
|
|
|
|
||||||
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
||||||
|
|
||||||
error: any use of this value will cause an error
|
error[E0080]: it is undefined behavior to use this value
|
||||||
--> $DIR/const-pointer-values-in-various-types.rs:54:5
|
--> $DIR/const-pointer-values-in-various-types.rs:54:5
|
||||||
|
|
|
|
||||||
LL | const I32_REF_I128_UNION: i128 = unsafe { Nonsense { int_32_ref: &3 }.int_128 };
|
LL | const I32_REF_I128_UNION: i128 = unsafe { Nonsense { int_32_ref: &3 }.int_128 };
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ attempted to read undefined bytes
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type validation failed: encountered uninitialized bytes, but expected initialized plain (non-pointer) bytes
|
||||||
|
|
|
||||||
|
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
||||||
|
|
||||||
error: any use of this value will cause an error
|
error: any use of this value will cause an error
|
||||||
--> $DIR/const-pointer-values-in-various-types.rs:57:5
|
--> $DIR/const-pointer-values-in-various-types.rs:57:5
|
||||||
|
@ -16,12 +16,12 @@ enum Enum {
|
|||||||
A = 0,
|
A = 0,
|
||||||
}
|
}
|
||||||
union TransmuteEnum {
|
union TransmuteEnum {
|
||||||
a: &'static u8,
|
in1: &'static u8,
|
||||||
out: Enum,
|
out1: Enum,
|
||||||
}
|
}
|
||||||
|
|
||||||
// A pointer is guaranteed non-null
|
// A pointer is guaranteed non-null
|
||||||
const BAD_ENUM: Enum = unsafe { TransmuteEnum { a: &1 }.out };
|
const BAD_ENUM: Enum = unsafe { TransmuteEnum { in1: &1 }.out1 };
|
||||||
//~^ ERROR is undefined behavior
|
//~^ ERROR is undefined behavior
|
||||||
|
|
||||||
// (Potentially) invalid enum discriminant
|
// (Potentially) invalid enum discriminant
|
||||||
@ -48,8 +48,8 @@ const BAD_ENUM3: Enum2 = unsafe { TransmuteEnum2 { in2: &0 }.out1 };
|
|||||||
const BAD_ENUM4: Wrap<Enum2> = unsafe { TransmuteEnum2 { in2: &0 }.out2 };
|
const BAD_ENUM4: Wrap<Enum2> = unsafe { TransmuteEnum2 { in2: &0 }.out2 };
|
||||||
//~^ ERROR is undefined behavior
|
//~^ ERROR is undefined behavior
|
||||||
|
|
||||||
// Undef enum discriminant. In an arry to avoid `Scalar` layout.
|
// Undef enum discriminant.
|
||||||
const BAD_ENUM_UNDEF: [Enum2; 2] = [unsafe { TransmuteEnum2 { in3: () }.out1 }; 2];
|
const BAD_ENUM_UNDEF : Enum2 = unsafe { TransmuteEnum2 { in3: () }.out1 };
|
||||||
//~^ ERROR is undefined behavior
|
//~^ ERROR is undefined behavior
|
||||||
|
|
||||||
// Pointer value in an enum with a niche that is not just 0.
|
// Pointer value in an enum with a niche that is not just 0.
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
error[E0080]: it is undefined behavior to use this value
|
error[E0080]: it is undefined behavior to use this value
|
||||||
--> $DIR/ub-enum.rs:24:1
|
--> $DIR/ub-enum.rs:24:1
|
||||||
|
|
|
|
||||||
LL | const BAD_ENUM: Enum = unsafe { TransmuteEnum { a: &1 }.out };
|
LL | const BAD_ENUM: Enum = unsafe { TransmuteEnum { in1: &1 }.out1 };
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type validation failed: encountered a pointer, but expected a valid enum discriminant
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type validation failed: encountered a pointer, but expected a valid enum discriminant
|
||||||
|
|
|
|
||||||
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
||||||
|
|
||||||
@ -33,8 +33,8 @@ LL | const BAD_ENUM4: Wrap<Enum2> = unsafe { TransmuteEnum2 { in2: &0 }.out2 };
|
|||||||
error[E0080]: it is undefined behavior to use this value
|
error[E0080]: it is undefined behavior to use this value
|
||||||
--> $DIR/ub-enum.rs:52:1
|
--> $DIR/ub-enum.rs:52:1
|
||||||
|
|
|
|
||||||
LL | const BAD_ENUM_UNDEF: [Enum2; 2] = [unsafe { TransmuteEnum2 { in3: () }.out1 }; 2];
|
LL | const BAD_ENUM_UNDEF : Enum2 = unsafe { TransmuteEnum2 { in3: () }.out1 };
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type validation failed: encountered uninitialized bytes at [0], but expected a valid enum discriminant
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type validation failed: encountered uninitialized bytes, but expected a valid enum discriminant
|
||||||
|
|
|
|
||||||
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
||||||
|
|
||||||
|
@ -34,7 +34,8 @@ const fn read_field2() -> Field2 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const fn read_field3() -> Field3 {
|
const fn read_field3() -> Field3 {
|
||||||
const FIELD3: Field3 = unsafe { UNION.field3 }; //~ ERROR any use of this value
|
const FIELD3: Field3 = unsafe { UNION.field3 };
|
||||||
|
//~^ ERROR it is undefined behavior to use this value
|
||||||
FIELD3
|
FIELD3
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
error: any use of this value will cause an error
|
error[E0080]: it is undefined behavior to use this value
|
||||||
--> $DIR/union-const-eval-field.rs:37:5
|
--> $DIR/union-const-eval-field.rs:37:5
|
||||||
|
|
|
|
||||||
LL | const FIELD3: Field3 = unsafe { UNION.field3 }; //~ ERROR any use of this value
|
LL | const FIELD3: Field3 = unsafe { UNION.field3 };
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ attempted to read undefined bytes
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type validation failed: encountered uninitialized bytes, but expected initialized plain (non-pointer) bytes
|
||||||
|
|
|
|
||||||
= note: #[deny(const_err)] on by default
|
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
||||||
|
|
||||||
error: aborting due to previous error
|
error: aborting due to previous error
|
||||||
|
|
||||||
|
For more information about this error, try `rustc --explain E0080`.
|
||||||
|
@ -20,7 +20,7 @@ union DummyUnion {
|
|||||||
|
|
||||||
const UNION: DummyUnion = DummyUnion { field1: 1065353216 };
|
const UNION: DummyUnion = DummyUnion { field1: 1065353216 };
|
||||||
|
|
||||||
const FIELD3: Field3 = unsafe { UNION.field3 }; //~ ERROR will cause an error
|
const FIELD3: Field3 = unsafe { UNION.field3 }; //~ ERROR it is undefined behavior to use this value
|
||||||
|
|
||||||
const FIELD_PATH: Struct = Struct { //~ ERROR it is undefined behavior to use this value
|
const FIELD_PATH: Struct = Struct { //~ ERROR it is undefined behavior to use this value
|
||||||
a: 42,
|
a: 42,
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
error: any use of this value will cause an error
|
error[E0080]: it is undefined behavior to use this value
|
||||||
--> $DIR/union-ice.rs:23:1
|
--> $DIR/union-ice.rs:23:1
|
||||||
|
|
|
|
||||||
LL | const FIELD3: Field3 = unsafe { UNION.field3 }; //~ ERROR will cause an error
|
LL | const FIELD3: Field3 = unsafe { UNION.field3 }; //~ ERROR it is undefined behavior to use this value
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ attempted to read undefined bytes
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type validation failed: encountered uninitialized bytes, but expected initialized plain (non-pointer) bytes
|
||||||
|
|
|
|
||||||
= note: #[deny(const_err)] on by default
|
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rust compiler repository if you believe it should not be considered undefined behavior
|
||||||
|
|
||||||
error[E0080]: it is undefined behavior to use this value
|
error[E0080]: it is undefined behavior to use this value
|
||||||
--> $DIR/union-ice.rs:25:1
|
--> $DIR/union-ice.rs:25:1
|
||||||
|
Loading…
Reference in New Issue
Block a user