Add new query just for static initializers

This commit is contained in:
Oli Scherer 2023-10-09 08:02:46 +00:00
parent fc9d1a8133
commit 95004e5ae2
13 changed files with 65 additions and 43 deletions

View file

@ -40,6 +40,13 @@ pub fn provide(providers: &mut Providers) {
const_eval::provide(providers); const_eval::provide(providers);
providers.eval_to_const_value_raw = const_eval::eval_to_const_value_raw_provider; providers.eval_to_const_value_raw = const_eval::eval_to_const_value_raw_provider;
providers.eval_to_allocation_raw = const_eval::eval_to_allocation_raw_provider; providers.eval_to_allocation_raw = const_eval::eval_to_allocation_raw_provider;
providers.eval_static_initializer_raw = |tcx, def_id| {
assert!(tcx.is_static(def_id));
let instance = ty::Instance::mono(tcx, def_id);
let gid = rustc_middle::mir::interpret::GlobalId { instance, promoted: None };
let param_env = ty::ParamEnv::reveal_all();
Ok(tcx.eval_to_allocation_raw(param_env.and(gid))?.alloc_id)
};
providers.hooks.const_caller_location = util::caller_location::const_caller_location_provider; providers.hooks.const_caller_location = util::caller_location::const_caller_location_provider;
providers.eval_to_valtree = |tcx, param_env_and_value| { providers.eval_to_valtree = |tcx, param_env_and_value| {
let (param_env, raw) = param_env_and_value.into_parts(); let (param_env, raw) = param_env_and_value.into_parts();

View file

@ -83,6 +83,7 @@ impl Into<ErrorGuaranteed> for ReportedErrorInfo {
TrivialTypeTraversalImpls! { ErrorHandled } TrivialTypeTraversalImpls! { ErrorHandled }
pub type EvalToAllocationRawResult<'tcx> = Result<ConstAlloc<'tcx>, ErrorHandled>; pub type EvalToAllocationRawResult<'tcx> = Result<ConstAlloc<'tcx>, ErrorHandled>;
pub type EvalStaticInitializerRawResult = Result<AllocId, ErrorHandled>;
pub type EvalToConstValueResult<'tcx> = Result<ConstValue<'tcx>, ErrorHandled>; pub type EvalToConstValueResult<'tcx> = Result<ConstValue<'tcx>, ErrorHandled>;
/// `Ok(None)` indicates the constant was fine, but the valtree couldn't be constructed. /// `Ok(None)` indicates the constant was fine, but the valtree couldn't be constructed.
/// This is needed in `thir::pattern::lower_inline_const`. /// This is needed in `thir::pattern::lower_inline_const`.

View file

@ -142,11 +142,12 @@ use crate::ty::GenericArgKind;
use crate::ty::{self, Instance, Ty, TyCtxt}; use crate::ty::{self, Instance, Ty, TyCtxt};
pub use self::error::{ pub use self::error::{
BadBytesAccess, CheckAlignMsg, CheckInAllocMsg, ErrorHandled, EvalToAllocationRawResult, BadBytesAccess, CheckAlignMsg, CheckInAllocMsg, ErrorHandled, EvalStaticInitializerRawResult,
EvalToConstValueResult, EvalToValTreeResult, ExpectedKind, InterpError, InterpErrorInfo, EvalToAllocationRawResult, EvalToConstValueResult, EvalToValTreeResult, ExpectedKind,
InterpResult, InvalidMetaKind, InvalidProgramInfo, MachineStopType, Misalignment, PointerKind, InterpError, InterpErrorInfo, InterpResult, InvalidMetaKind, InvalidProgramInfo,
ReportedErrorInfo, ResourceExhaustionInfo, ScalarSizeMismatch, UndefinedBehaviorInfo, MachineStopType, Misalignment, PointerKind, ReportedErrorInfo, ResourceExhaustionInfo,
UnsupportedOpInfo, ValidationErrorInfo, ValidationErrorKind, ScalarSizeMismatch, UndefinedBehaviorInfo, UnsupportedOpInfo, ValidationErrorInfo,
ValidationErrorKind,
}; };
pub use self::value::Scalar; pub use self::value::Scalar;

View file

@ -194,22 +194,8 @@ impl<'tcx> TyCtxtAt<'tcx> {
) -> Result<mir::ConstAllocation<'tcx>, ErrorHandled> { ) -> Result<mir::ConstAllocation<'tcx>, ErrorHandled> {
trace!("eval_static_initializer: Need to compute {:?}", def_id); trace!("eval_static_initializer: Need to compute {:?}", def_id);
assert!(self.is_static(def_id)); assert!(self.is_static(def_id));
let instance = ty::Instance::mono(*self, def_id); let alloc_id = self.eval_static_initializer_raw(def_id)?;
let gid = GlobalId { instance, promoted: None }; Ok(self.global_alloc(alloc_id).unwrap_memory())
self.eval_to_allocation(gid, ty::ParamEnv::reveal_all())
}
/// Evaluate anything constant-like, returning the allocation of the final memory.
///
/// The span is entirely ignored here, but still helpful for better query cycle errors.
fn eval_to_allocation(
self,
gid: GlobalId<'tcx>,
param_env: ty::ParamEnv<'tcx>,
) -> Result<mir::ConstAllocation<'tcx>, ErrorHandled> {
trace!("eval_to_allocation: Need to compute {:?}", gid);
let raw_const = self.eval_to_allocation_raw(param_env.and(gid))?;
Ok(self.global_alloc(raw_const.alloc_id).unwrap_memory())
} }
} }
@ -237,10 +223,6 @@ impl<'tcx> TyCtxtEnsure<'tcx> {
pub fn eval_static_initializer(self, def_id: DefId) { pub fn eval_static_initializer(self, def_id: DefId) {
trace!("eval_static_initializer: Need to compute {:?}", def_id); trace!("eval_static_initializer: Need to compute {:?}", def_id);
assert!(self.tcx.is_static(def_id)); assert!(self.tcx.is_static(def_id));
let instance = ty::Instance::mono(self.tcx, def_id); self.eval_static_initializer_raw(def_id);
let gid = GlobalId { instance, promoted: None };
let param_env = ty::ParamEnv::reveal_all();
trace!("eval_to_allocation: Need to compute {:?}", gid);
self.eval_to_allocation_raw(param_env.and(gid))
} }
} }

View file

@ -277,6 +277,7 @@ trivial! {
rustc_middle::mir::interpret::CtfeProvenance, rustc_middle::mir::interpret::CtfeProvenance,
rustc_middle::mir::interpret::ErrorHandled, rustc_middle::mir::interpret::ErrorHandled,
rustc_middle::mir::interpret::LitToConstError, rustc_middle::mir::interpret::LitToConstError,
rustc_middle::mir::interpret::EvalStaticInitializerRawResult,
rustc_middle::thir::ExprId, rustc_middle::thir::ExprId,
rustc_middle::traits::CodegenObligationError, rustc_middle::traits::CodegenObligationError,
rustc_middle::traits::EvaluationResult, rustc_middle::traits::EvaluationResult,

View file

@ -20,7 +20,8 @@ use crate::middle::stability::{self, DeprecationEntry};
use crate::mir; use crate::mir;
use crate::mir::interpret::GlobalId; use crate::mir::interpret::GlobalId;
use crate::mir::interpret::{ use crate::mir::interpret::{
EvalToAllocationRawResult, EvalToConstValueResult, EvalToValTreeResult, EvalStaticInitializerRawResult, EvalToAllocationRawResult, EvalToConstValueResult,
EvalToValTreeResult,
}; };
use crate::mir::interpret::{LitToConstError, LitToConstInput}; use crate::mir::interpret::{LitToConstError, LitToConstInput};
use crate::mir::mono::CodegenUnit; use crate::mir::mono::CodegenUnit;
@ -1061,7 +1062,7 @@ rustc_queries! {
/// Evaluates a constant and returns the computed allocation. /// Evaluates a constant and returns the computed allocation.
/// ///
/// **Do not use this** directly, use the `tcx.eval_static_initializer` wrapper. /// **Do not use this** directly, use the `eval_to_const_value` or `eval_to_valtree` instead.
query eval_to_allocation_raw(key: ty::ParamEnvAnd<'tcx, GlobalId<'tcx>>) query eval_to_allocation_raw(key: ty::ParamEnvAnd<'tcx, GlobalId<'tcx>>)
-> EvalToAllocationRawResult<'tcx> { -> EvalToAllocationRawResult<'tcx> {
desc { |tcx| desc { |tcx|
@ -1071,6 +1072,15 @@ rustc_queries! {
cache_on_disk_if { true } cache_on_disk_if { true }
} }
/// Evaluate a static's initializer, returning the allocation of the initializer's memory.
query eval_static_initializer_raw(key: DefId) -> EvalStaticInitializerRawResult {
desc { |tcx|
"evaluating initializer of static `{}`",
tcx.def_path_str(key)
}
cache_on_disk_if { key.is_local() }
}
/// Evaluates const items or anonymous constants /// Evaluates const items or anonymous constants
/// (such as enum variant explicit discriminants or array lengths) /// (such as enum variant explicit discriminants or array lengths)
/// into a representation suitable for the type system and const generics. /// into a representation suitable for the type system and const generics.

View file

@ -1,10 +1,15 @@
error[E0391]: cycle detected when const-evaluating + checking `FOO` error[E0391]: cycle detected when evaluating initializer of static `FOO`
--> $DIR/recursive-zst-static.rs:10:1
|
LL | static FOO: () = FOO;
| ^^^^^^^^^^^^^^
|
note: ...which requires const-evaluating + checking `FOO`...
--> $DIR/recursive-zst-static.rs:10:18 --> $DIR/recursive-zst-static.rs:10:18
| |
LL | static FOO: () = FOO; LL | static FOO: () = FOO;
| ^^^ | ^^^
| = note: ...which again requires evaluating initializer of static `FOO`, completing the cycle
= note: ...which immediately requires const-evaluating + checking `FOO` again
note: cycle used when linting top-level module note: cycle used when linting top-level module
--> $DIR/recursive-zst-static.rs:10:1 --> $DIR/recursive-zst-static.rs:10:1
| |

View file

@ -7,7 +7,7 @@
// can depend on this fact and will thus do unsound things when it is violated. // can depend on this fact and will thus do unsound things when it is violated.
// See https://github.com/rust-lang/rust/issues/71078 for more details. // See https://github.com/rust-lang/rust/issues/71078 for more details.
static FOO: () = FOO; //~ cycle detected when const-evaluating + checking `FOO` static FOO: () = FOO; //~ cycle detected when evaluating initializer of static `FOO`
fn main() { fn main() {
FOO FOO

View file

@ -1,10 +1,15 @@
error[E0391]: cycle detected when const-evaluating + checking `FOO` error[E0391]: cycle detected when evaluating initializer of static `FOO`
--> $DIR/recursive-zst-static.rs:10:1
|
LL | static FOO: () = FOO;
| ^^^^^^^^^^^^^^
|
note: ...which requires const-evaluating + checking `FOO`...
--> $DIR/recursive-zst-static.rs:10:18 --> $DIR/recursive-zst-static.rs:10:18
| |
LL | static FOO: () = FOO; LL | static FOO: () = FOO;
| ^^^ | ^^^
| = note: ...which again requires evaluating initializer of static `FOO`, completing the cycle
= note: ...which immediately requires const-evaluating + checking `FOO` again
note: cycle used when linting top-level module note: cycle used when linting top-level module
--> $DIR/recursive-zst-static.rs:10:1 --> $DIR/recursive-zst-static.rs:10:1
| |

View file

@ -4,13 +4,18 @@ error[E0080]: could not evaluate static initializer
LL | pub static mut B: () = unsafe { A = 1; }; LL | pub static mut B: () = unsafe { A = 1; };
| ^^^^^ modifying a static's initial value from another static's initializer | ^^^^^ modifying a static's initial value from another static's initializer
error[E0391]: cycle detected when const-evaluating + checking `C` error[E0391]: cycle detected when evaluating initializer of static `C`
--> $DIR/write-to-static-mut-in-static.rs:5:1
|
LL | pub static mut C: u32 = unsafe { C = 1; 0 };
| ^^^^^^^^^^^^^^^^^^^^^
|
note: ...which requires const-evaluating + checking `C`...
--> $DIR/write-to-static-mut-in-static.rs:5:34 --> $DIR/write-to-static-mut-in-static.rs:5:34
| |
LL | pub static mut C: u32 = unsafe { C = 1; 0 }; LL | pub static mut C: u32 = unsafe { C = 1; 0 };
| ^^^^^ | ^^^^^
| = note: ...which again requires evaluating initializer of static `C`, completing the cycle
= note: ...which immediately requires const-evaluating + checking `C` again
note: cycle used when linting top-level module note: cycle used when linting top-level module
--> $DIR/write-to-static-mut-in-static.rs:1:1 --> $DIR/write-to-static-mut-in-static.rs:1:1
| |

View file

@ -1,4 +1,4 @@
pub static FOO: u32 = FOO; pub static FOO: u32 = FOO;
//~^ ERROR cycle detected when const-evaluating + checking `FOO` //~^ ERROR cycle detected when evaluating initializer of static `FOO`
fn main() {} fn main() {}

View file

@ -1,10 +1,15 @@
error[E0391]: cycle detected when const-evaluating + checking `FOO` error[E0391]: cycle detected when evaluating initializer of static `FOO`
--> $DIR/recursive-static-definition.rs:1:1
|
LL | pub static FOO: u32 = FOO;
| ^^^^^^^^^^^^^^^^^^^
|
note: ...which requires const-evaluating + checking `FOO`...
--> $DIR/recursive-static-definition.rs:1:23 --> $DIR/recursive-static-definition.rs:1:23
| |
LL | pub static FOO: u32 = FOO; LL | pub static FOO: u32 = FOO;
| ^^^ | ^^^
| = note: ...which again requires evaluating initializer of static `FOO`, completing the cycle
= note: ...which immediately requires const-evaluating + checking `FOO` again
note: cycle used when linting top-level module note: cycle used when linting top-level module
--> $DIR/recursive-static-definition.rs:1:1 --> $DIR/recursive-static-definition.rs:1:1
| |

View file

@ -8,5 +8,5 @@ error: the compiler unexpectedly panicked. this is a bug.
query stack during panic: query stack during panic:
#0 [eval_to_allocation_raw] const-evaluating + checking `C` #0 [eval_to_allocation_raw] const-evaluating + checking `C`
#1 [lint_mod] linting top-level module #1 [eval_static_initializer_raw] evaluating initializer of static `C`
end of query stack end of query stack