rust/crates/hir-ty/src/consteval.rs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

308 lines
9.8 KiB
Rust
Raw Normal View History

//! Constant evaluation details
2023-02-03 14:46:25 +03:30
use base_db::CrateId;
use chalk_ir::{cast::Cast, BoundVar, DebruijnIndex};
use hir_def::{
hir::Expr,
2023-03-08 20:58:52 +03:30
path::Path,
2023-02-03 14:46:25 +03:30
resolver::{Resolver, ValueNs},
2023-06-05 14:57:19 +03:30
type_ref::LiteralConstRef,
ConstBlockLoc, EnumVariantId, GeneralConstId, StaticId,
};
2023-02-03 14:46:25 +03:30
use la_arena::{Idx, RawIdx};
2022-03-09 22:20:24 +03:30
use stdx::never;
2023-05-12 18:17:15 +03:30
use triomphe::Arc;
2022-03-09 22:20:24 +03:30
use crate::{
2023-05-26 00:45:37 +03:30
db::HirDatabase, infer::InferenceContext, lower::ParamLoweringMode,
mir::monomorphize_mir_body_bad, to_placeholder_idx, utils::Generics, Const, ConstData,
ConstScalar, ConstValue, GenericArg, Interner, MemoryMap, Substitution, TraitEnvironment, Ty,
TyBuilder,
2022-03-09 22:20:24 +03:30
};
2023-02-03 14:46:25 +03:30
use super::mir::{interpret_mir, lower_to_mir, pad16, MirEvalError, MirLowerError};
/// Extension trait for [`Const`]
pub trait ConstExt {
/// Is a [`Const`] unknown?
fn is_unknown(&self) -> bool;
}
impl ConstExt for Const {
fn is_unknown(&self) -> bool {
2021-12-19 18:58:39 +02:00
match self.data(Interner).value {
// interned Unknown
chalk_ir::ConstValue::Concrete(chalk_ir::ConcreteConst {
interned: ConstScalar::Unknown,
}) => true,
// interned concrete anything else
chalk_ir::ConstValue::Concrete(..) => false,
_ => {
2021-08-15 20:46:13 +08:00
tracing::error!(
"is_unknown was called on a non-concrete constant value! {:?}",
self
);
true
}
}
}
}
2022-03-20 17:15:28 +03:30
#[derive(Debug, Clone, PartialEq, Eq)]
2021-12-05 01:51:36 +03:30
pub enum ConstEvalError {
2023-02-03 14:46:25 +03:30
MirLowerError(MirLowerError),
MirEvalError(MirEvalError),
2021-12-05 01:51:36 +03:30
}
2023-02-03 14:46:25 +03:30
impl From<MirLowerError> for ConstEvalError {
fn from(value: MirLowerError) -> Self {
match value {
2023-04-28 20:44:30 +03:30
MirLowerError::ConstEvalError(_, e) => *e,
2023-02-03 14:46:25 +03:30
_ => ConstEvalError::MirLowerError(value),
2021-12-05 01:51:36 +03:30
}
}
}
2023-02-03 14:46:25 +03:30
impl From<MirEvalError> for ConstEvalError {
fn from(value: MirEvalError) -> Self {
ConstEvalError::MirEvalError(value)
2021-12-05 01:51:36 +03:30
}
}
2022-03-09 22:20:24 +03:30
pub(crate) fn path_to_const(
db: &dyn HirDatabase,
resolver: &Resolver,
2023-03-08 20:58:52 +03:30
path: &Path,
2022-03-09 22:20:24 +03:30
mode: ParamLoweringMode,
args_lazy: impl FnOnce() -> Generics,
debruijn: DebruijnIndex,
expected_ty: Ty,
2022-03-09 22:20:24 +03:30
) -> Option<Const> {
2022-12-30 08:05:03 +00:00
match resolver.resolve_path_in_value_ns_fully(db.upcast(), path) {
2022-03-09 22:20:24 +03:30
Some(ValueNs::GenericParam(p)) => {
let ty = db.const_param_ty(p);
let args = args_lazy();
let value = match mode {
ParamLoweringMode::Placeholder => {
ConstValue::Placeholder(to_placeholder_idx(db, p.into()))
}
ParamLoweringMode::Variable => match args.param_idx(p.into()) {
2023-07-06 17:33:17 +03:30
Some(it) => ConstValue::BoundVar(BoundVar::new(debruijn, it)),
2022-03-09 22:20:24 +03:30
None => {
never!(
2023-03-08 20:58:52 +03:30
"Generic list doesn't contain this param: {:?}, {:?}, {:?}",
2022-03-09 22:20:24 +03:30
args,
path,
p
);
return None;
}
},
};
Some(ConstData { ty, value }.intern(Interner))
}
Some(ValueNs::ConstId(c)) => Some(intern_const_scalar(
ConstScalar::UnevaluatedConst(c.into(), Substitution::empty(Interner)),
expected_ty,
)),
2022-03-09 22:20:24 +03:30
_ => None,
}
}
pub fn unknown_const(ty: Ty) -> Const {
ConstData {
ty,
value: ConstValue::Concrete(chalk_ir::ConcreteConst { interned: ConstScalar::Unknown }),
}
.intern(Interner)
}
pub fn unknown_const_as_generic(ty: Ty) -> GenericArg {
unknown_const(ty).cast(Interner)
2022-03-09 22:20:24 +03:30
}
2022-04-07 05:30:33 +04:30
/// Interns a constant scalar with the given type
2022-07-17 18:22:11 +03:00
pub fn intern_const_scalar(value: ConstScalar, ty: Ty) -> Const {
2022-04-07 05:30:33 +04:30
ConstData { ty, value: ConstValue::Concrete(chalk_ir::ConcreteConst { interned: value }) }
.intern(Interner)
}
2023-02-03 14:46:25 +03:30
/// Interns a constant scalar with the given type
2023-06-05 14:57:19 +03:30
pub fn intern_const_ref(
db: &dyn HirDatabase,
value: &LiteralConstRef,
ty: Ty,
krate: CrateId,
) -> Const {
let layout = db.layout_of_ty(ty.clone(), Arc::new(TraitEnvironment::empty(krate)));
2023-02-03 14:46:25 +03:30
let bytes = match value {
2023-06-05 14:57:19 +03:30
LiteralConstRef::Int(i) => {
2023-02-03 14:46:25 +03:30
// FIXME: We should handle failure of layout better.
2023-07-06 17:33:17 +03:30
let size = layout.map(|it| it.size.bytes_usize()).unwrap_or(16);
2023-02-03 14:46:25 +03:30
ConstScalar::Bytes(i.to_le_bytes()[0..size].to_vec(), MemoryMap::default())
}
2023-06-05 14:57:19 +03:30
LiteralConstRef::UInt(i) => {
2023-07-06 17:33:17 +03:30
let size = layout.map(|it| it.size.bytes_usize()).unwrap_or(16);
2023-02-03 14:46:25 +03:30
ConstScalar::Bytes(i.to_le_bytes()[0..size].to_vec(), MemoryMap::default())
}
2023-06-05 14:57:19 +03:30
LiteralConstRef::Bool(b) => ConstScalar::Bytes(vec![*b as u8], MemoryMap::default()),
LiteralConstRef::Char(c) => {
2023-02-03 14:46:25 +03:30
ConstScalar::Bytes((*c as u32).to_le_bytes().to_vec(), MemoryMap::default())
}
2023-06-05 14:57:19 +03:30
LiteralConstRef::Unknown => ConstScalar::Unknown,
2023-02-03 14:46:25 +03:30
};
intern_const_scalar(bytes, ty)
}
/// Interns a possibly-unknown target usize
2023-02-03 14:46:25 +03:30
pub fn usize_const(db: &dyn HirDatabase, value: Option<u128>, krate: CrateId) -> Const {
intern_const_ref(
db,
2023-06-05 14:57:19 +03:30
&value.map_or(LiteralConstRef::Unknown, LiteralConstRef::UInt),
2023-02-03 14:46:25 +03:30
TyBuilder::usize(),
krate,
)
}
2023-05-12 18:17:15 +03:30
pub fn try_const_usize(db: &dyn HirDatabase, c: &Const) -> Option<u128> {
2023-02-03 14:46:25 +03:30
match &c.data(Interner).value {
chalk_ir::ConstValue::BoundVar(_) => None,
chalk_ir::ConstValue::InferenceVar(_) => None,
chalk_ir::ConstValue::Placeholder(_) => None,
chalk_ir::ConstValue::Concrete(c) => match &c.interned {
2023-07-06 17:33:17 +03:30
ConstScalar::Bytes(it, _) => Some(u128::from_le_bytes(pad16(&it, false))),
2023-05-12 18:17:15 +03:30
ConstScalar::UnevaluatedConst(c, subst) => {
let ec = db.const_eval(*c, subst.clone(), None).ok()?;
2023-05-12 18:17:15 +03:30
try_const_usize(db, &ec)
}
2023-02-03 14:46:25 +03:30
_ => None,
},
}
}
2022-03-20 17:15:28 +03:30
pub(crate) fn const_eval_recover(
_: &dyn HirDatabase,
_: &[String],
2023-05-12 18:17:15 +03:30
_: &GeneralConstId,
_: &Substitution,
_: &Option<Arc<TraitEnvironment>>,
2023-02-03 14:46:25 +03:30
) -> Result<Const, ConstEvalError> {
Err(ConstEvalError::MirLowerError(MirLowerError::Loop))
2022-03-20 17:15:28 +03:30
}
2023-05-12 18:17:15 +03:30
pub(crate) fn const_eval_static_recover(
_: &dyn HirDatabase,
_: &[String],
_: &StaticId,
) -> Result<Const, ConstEvalError> {
Err(ConstEvalError::MirLowerError(MirLowerError::Loop))
}
2023-02-03 14:46:25 +03:30
pub(crate) fn const_eval_discriminant_recover(
2022-08-06 18:50:21 +02:00
_: &dyn HirDatabase,
_: &[String],
_: &EnumVariantId,
2023-02-03 14:46:25 +03:30
) -> Result<i128, ConstEvalError> {
Err(ConstEvalError::MirLowerError(MirLowerError::Loop))
2022-08-06 18:50:21 +02:00
}
2023-02-03 14:46:25 +03:30
pub(crate) fn const_eval_query(
2022-03-20 17:15:28 +03:30
db: &dyn HirDatabase,
2023-05-12 18:17:15 +03:30
def: GeneralConstId,
subst: Substitution,
trait_env: Option<Arc<TraitEnvironment>>,
2023-02-03 14:46:25 +03:30
) -> Result<Const, ConstEvalError> {
2023-05-12 18:17:15 +03:30
let body = match def {
2023-05-26 00:45:37 +03:30
GeneralConstId::ConstId(c) => {
db.monomorphized_mir_body(c.into(), subst, db.trait_environment(c.into()))?
}
2023-06-12 00:37:11 +03:30
GeneralConstId::ConstBlockId(c) => {
let ConstBlockLoc { parent, root } = db.lookup_intern_anonymous_const(c);
let body = db.body(parent);
let infer = db.infer(parent);
2023-05-26 00:45:37 +03:30
Arc::new(monomorphize_mir_body_bad(
db,
lower_to_mir(db, parent, &body, &infer, root)?,
2023-05-26 00:45:37 +03:30
subst,
db.trait_environment_for_body(parent),
2023-05-26 00:45:37 +03:30
)?)
2023-05-12 18:17:15 +03:30
}
2023-06-05 14:57:19 +03:30
GeneralConstId::InTypeConstId(c) => db.mir_body(c.into())?,
2023-05-12 18:17:15 +03:30
};
let c = interpret_mir(db, body, false, trait_env).0?;
2023-05-12 18:17:15 +03:30
Ok(c)
}
pub(crate) fn const_eval_static_query(
db: &dyn HirDatabase,
def: StaticId,
) -> Result<Const, ConstEvalError> {
2023-05-26 00:45:37 +03:30
let body = db.monomorphized_mir_body(
def.into(),
Substitution::empty(Interner),
db.trait_environment_for_body(def.into()),
)?;
let c = interpret_mir(db, body, false, None).0?;
2023-02-03 14:46:25 +03:30
Ok(c)
2022-03-20 17:15:28 +03:30
}
2023-02-03 14:46:25 +03:30
pub(crate) fn const_eval_discriminant_variant(
2022-08-06 18:50:21 +02:00
db: &dyn HirDatabase,
variant_id: EnumVariantId,
2023-02-03 14:46:25 +03:30
) -> Result<i128, ConstEvalError> {
2022-08-06 18:50:21 +02:00
let def = variant_id.into();
let body = db.body(def);
2023-02-03 14:46:25 +03:30
if body.exprs[body.body_expr] == Expr::Missing {
let prev_idx: u32 = variant_id.local_id.into_raw().into();
let prev_idx = prev_idx.checked_sub(1).map(RawIdx::from).map(Idx::from_raw);
let value = match prev_idx {
Some(local_id) => {
let prev_variant = EnumVariantId { local_id, parent: variant_id.parent };
1 + db.const_eval_discriminant(prev_variant)?
}
_ => 0,
};
return Ok(value);
}
2023-05-26 00:45:37 +03:30
let mir_body = db.monomorphized_mir_body(
def,
Substitution::empty(Interner),
db.trait_environment_for_body(def),
)?;
let c = interpret_mir(db, mir_body, false, None).0?;
2023-05-12 18:17:15 +03:30
let c = try_const_usize(db, &c).unwrap() as i128;
2023-02-03 14:46:25 +03:30
Ok(c)
2022-08-06 18:50:21 +02:00
}
2023-02-03 14:46:25 +03:30
// FIXME: Ideally constants in const eval should have separate body (issue #7434), and this function should
// get an `InferenceResult` instead of an `InferenceContext`. And we should remove `ctx.clone().resolve_all()` here
// and make this function private. See the fixme comment on `InferenceContext::resolve_all`.
pub(crate) fn eval_to_const(
2022-03-09 22:20:24 +03:30
expr: Idx<Expr>,
mode: ParamLoweringMode,
ctx: &mut InferenceContext<'_>,
2022-03-09 22:20:24 +03:30
args: impl FnOnce() -> Generics,
debruijn: DebruijnIndex,
) -> Const {
2023-02-03 14:46:25 +03:30
let db = ctx.db;
let infer = ctx.clone().resolve_all();
2022-03-09 22:20:24 +03:30
if let Expr::Path(p) = &ctx.body.exprs[expr] {
let resolver = &ctx.resolver;
if let Some(c) = path_to_const(db, resolver, p, mode, args, debruijn, infer[expr].clone()) {
2022-03-09 22:20:24 +03:30
return c;
}
}
2023-02-03 14:46:25 +03:30
let infer = ctx.clone().resolve_all();
if let Ok(mir_body) = lower_to_mir(ctx.db, ctx.owner, &ctx.body, &infer, expr) {
if let Ok(result) = interpret_mir(db, Arc::new(mir_body), true, None).0 {
2023-02-03 14:46:25 +03:30
return result;
}
}
unknown_const(infer[expr].clone())
2022-03-09 22:20:24 +03:30
}
2022-03-24 13:06:27 +04:30
#[cfg(test)]
mod tests;