2021-05-14 07:59:30 +00:00
|
|
|
//! Constant evaluation details
|
|
|
|
|
2021-12-04 22:21:36 +00:00
|
|
|
use std::{collections::HashMap, convert::TryInto, fmt::Display};
|
2021-05-14 07:59:30 +00:00
|
|
|
|
2021-12-04 22:21:36 +00:00
|
|
|
use chalk_ir::{IntTy, Scalar};
|
2021-05-14 07:59:30 +00:00
|
|
|
use hir_def::{
|
|
|
|
builtin_type::BuiltinUint,
|
2021-12-04 22:21:36 +00:00
|
|
|
expr::{ArithOp, BinaryOp, Expr, Literal, Pat},
|
2021-05-14 07:59:30 +00:00
|
|
|
type_ref::ConstScalar,
|
|
|
|
};
|
2021-12-04 22:21:36 +00:00
|
|
|
use hir_expand::name::Name;
|
|
|
|
use la_arena::Arena;
|
2021-05-14 07:59:30 +00:00
|
|
|
|
2021-12-04 22:21:36 +00:00
|
|
|
use crate::{Const, ConstData, ConstValue, InferenceResult, Interner, TyKind};
|
2021-05-14 07:59:30 +00:00
|
|
|
|
|
|
|
/// Extension trait for [`Const`]
|
2021-05-16 01:51:18 +00:00
|
|
|
pub trait ConstExt {
|
2021-05-14 07:59:30 +00:00
|
|
|
/// Is a [`Const`] unknown?
|
|
|
|
fn is_unknown(&self) -> bool;
|
|
|
|
}
|
|
|
|
|
2021-05-16 01:51:18 +00:00
|
|
|
impl ConstExt for Const {
|
2021-05-14 07:59:30 +00:00
|
|
|
fn is_unknown(&self) -> bool {
|
2021-12-19 16:58:39 +00:00
|
|
|
match self.data(Interner).value {
|
2021-05-14 07:59:30 +00:00
|
|
|
// 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 12:46:13 +00:00
|
|
|
tracing::error!(
|
|
|
|
"is_unknown was called on a non-concrete constant value! {:?}",
|
|
|
|
self
|
|
|
|
);
|
2021-05-14 07:59:30 +00:00
|
|
|
true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-04 22:21:36 +00:00
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct ConstEvalCtx<'a> {
|
|
|
|
pub exprs: &'a Arena<Expr>,
|
|
|
|
pub pats: &'a Arena<Pat>,
|
|
|
|
pub local_data: HashMap<Name, ComputedExpr>,
|
|
|
|
pub infer: &'a InferenceResult,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub enum ConstEvalError {
|
|
|
|
NotSupported(&'static str),
|
|
|
|
TypeError,
|
|
|
|
IncompleteExpr,
|
|
|
|
Panic(String),
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub enum ComputedExpr {
|
|
|
|
Literal(Literal),
|
|
|
|
Tuple(Box<[ComputedExpr]>),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Display for ComputedExpr {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
|
|
match self {
|
|
|
|
ComputedExpr::Literal(l) => match l {
|
|
|
|
Literal::Int(x, _) => {
|
|
|
|
if *x >= 16 {
|
|
|
|
write!(f, "{} ({:#X})", x, x)
|
|
|
|
} else {
|
|
|
|
write!(f, "{}", x)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Literal::Uint(x, _) => {
|
|
|
|
if *x >= 16 {
|
|
|
|
write!(f, "{} ({:#X})", x, x)
|
|
|
|
} else {
|
|
|
|
write!(f, "{}", x)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Literal::Float(x, _) => write!(f, "{}", x),
|
|
|
|
Literal::Bool(x) => write!(f, "{}", x),
|
|
|
|
Literal::Char(x) => write!(f, "{:?}", x),
|
|
|
|
Literal::String(x) => write!(f, "{:?}", x),
|
|
|
|
Literal::ByteString(x) => write!(f, "{:?}", x),
|
|
|
|
},
|
|
|
|
ComputedExpr::Tuple(t) => {
|
|
|
|
write!(f, "(")?;
|
|
|
|
for x in &**t {
|
|
|
|
write!(f, "{}, ", x)?;
|
|
|
|
}
|
|
|
|
write!(f, ")")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn scalar_max(scalar: &Scalar) -> i128 {
|
|
|
|
match scalar {
|
|
|
|
Scalar::Bool => 1,
|
|
|
|
Scalar::Char => u32::MAX as i128,
|
|
|
|
Scalar::Int(x) => match x {
|
|
|
|
IntTy::Isize => isize::MAX as i128,
|
|
|
|
IntTy::I8 => i8::MAX as i128,
|
|
|
|
IntTy::I16 => i16::MAX as i128,
|
|
|
|
IntTy::I32 => i32::MAX as i128,
|
|
|
|
IntTy::I64 => i64::MAX as i128,
|
|
|
|
IntTy::I128 => i128::MAX as i128,
|
|
|
|
},
|
|
|
|
Scalar::Uint(x) => match x {
|
|
|
|
chalk_ir::UintTy::Usize => usize::MAX as i128,
|
|
|
|
chalk_ir::UintTy::U8 => u8::MAX as i128,
|
|
|
|
chalk_ir::UintTy::U16 => u16::MAX as i128,
|
|
|
|
chalk_ir::UintTy::U32 => u32::MAX as i128,
|
|
|
|
chalk_ir::UintTy::U64 => u64::MAX as i128,
|
|
|
|
chalk_ir::UintTy::U128 => i128::MAX as i128, // ignore too big u128 for now
|
|
|
|
},
|
|
|
|
Scalar::Float(_) => 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn is_valid(scalar: &Scalar, value: i128) -> bool {
|
|
|
|
if value < 0 {
|
|
|
|
!matches!(scalar, Scalar::Uint(_)) && -scalar_max(scalar) - 1 <= value
|
|
|
|
} else {
|
|
|
|
value <= scalar_max(scalar)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn eval_const(expr: &Expr, mut ctx: ConstEvalCtx<'_>) -> Result<ComputedExpr, ConstEvalError> {
|
|
|
|
match expr {
|
|
|
|
Expr::Literal(l) => Ok(ComputedExpr::Literal(l.clone())),
|
|
|
|
&Expr::UnaryOp { expr, op } => {
|
|
|
|
let ty = &ctx.infer[expr];
|
|
|
|
let ev = eval_const(&ctx.exprs[expr], ctx)?;
|
|
|
|
match op {
|
|
|
|
hir_def::expr::UnaryOp::Deref => Err(ConstEvalError::NotSupported("deref")),
|
|
|
|
hir_def::expr::UnaryOp::Not => {
|
|
|
|
let v = match ev {
|
|
|
|
ComputedExpr::Literal(Literal::Bool(b)) => {
|
|
|
|
return Ok(ComputedExpr::Literal(Literal::Bool(!b)))
|
|
|
|
}
|
|
|
|
ComputedExpr::Literal(Literal::Int(v, _)) => v,
|
|
|
|
ComputedExpr::Literal(Literal::Uint(v, _)) => v
|
|
|
|
.try_into()
|
|
|
|
.map_err(|_| ConstEvalError::NotSupported("too big u128"))?,
|
|
|
|
_ => return Err(ConstEvalError::NotSupported("this kind of operator")),
|
|
|
|
};
|
|
|
|
let r = match ty.kind(Interner) {
|
|
|
|
TyKind::Scalar(Scalar::Uint(x)) => match x {
|
|
|
|
chalk_ir::UintTy::U8 => !(v as u8) as i128,
|
|
|
|
chalk_ir::UintTy::U16 => !(v as u16) as i128,
|
|
|
|
chalk_ir::UintTy::U32 => !(v as u32) as i128,
|
|
|
|
chalk_ir::UintTy::U64 => !(v as u64) as i128,
|
|
|
|
chalk_ir::UintTy::U128 => {
|
|
|
|
return Err(ConstEvalError::NotSupported("negation of u128"))
|
|
|
|
}
|
|
|
|
chalk_ir::UintTy::Usize => !(v as usize) as i128,
|
|
|
|
},
|
|
|
|
TyKind::Scalar(Scalar::Int(x)) => match x {
|
|
|
|
chalk_ir::IntTy::I8 => !(v as i8) as i128,
|
|
|
|
chalk_ir::IntTy::I16 => !(v as i16) as i128,
|
|
|
|
chalk_ir::IntTy::I32 => !(v as i32) as i128,
|
|
|
|
chalk_ir::IntTy::I64 => !(v as i64) as i128,
|
|
|
|
chalk_ir::IntTy::I128 => !v,
|
|
|
|
chalk_ir::IntTy::Isize => !(v as isize) as i128,
|
|
|
|
},
|
|
|
|
_ => return Err(ConstEvalError::NotSupported("unreachable?")),
|
|
|
|
};
|
|
|
|
Ok(ComputedExpr::Literal(Literal::Int(r, None)))
|
|
|
|
}
|
|
|
|
hir_def::expr::UnaryOp::Neg => {
|
|
|
|
let v = match ev {
|
|
|
|
ComputedExpr::Literal(Literal::Int(v, _)) => v,
|
|
|
|
ComputedExpr::Literal(Literal::Uint(v, _)) => v
|
|
|
|
.try_into()
|
|
|
|
.map_err(|_| ConstEvalError::NotSupported("too big u128"))?,
|
|
|
|
_ => return Err(ConstEvalError::NotSupported("this kind of operator")),
|
|
|
|
};
|
|
|
|
Ok(ComputedExpr::Literal(Literal::Int(
|
|
|
|
v.checked_neg().ok_or_else(|| {
|
|
|
|
ConstEvalError::Panic("overflow in negation".to_string())
|
|
|
|
})?,
|
|
|
|
None,
|
|
|
|
)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
&Expr::BinaryOp { lhs, rhs, op } => {
|
|
|
|
let ty = &ctx.infer[lhs];
|
|
|
|
let lhs = eval_const(&ctx.exprs[lhs], ctx.clone())?;
|
|
|
|
let rhs = eval_const(&ctx.exprs[rhs], ctx.clone())?;
|
|
|
|
let op = op.ok_or(ConstEvalError::IncompleteExpr)?;
|
|
|
|
let v1 = match lhs {
|
|
|
|
ComputedExpr::Literal(Literal::Int(v, _)) => v,
|
|
|
|
ComputedExpr::Literal(Literal::Uint(v, _)) => {
|
|
|
|
v.try_into().map_err(|_| ConstEvalError::NotSupported("too big u128"))?
|
|
|
|
}
|
|
|
|
_ => return Err(ConstEvalError::NotSupported("this kind of operator")),
|
|
|
|
};
|
|
|
|
let v2 = match rhs {
|
|
|
|
ComputedExpr::Literal(Literal::Int(v, _)) => v,
|
|
|
|
ComputedExpr::Literal(Literal::Uint(v, _)) => {
|
|
|
|
v.try_into().map_err(|_| ConstEvalError::NotSupported("too big u128"))?
|
|
|
|
}
|
|
|
|
_ => return Err(ConstEvalError::NotSupported("this kind of operator")),
|
|
|
|
};
|
|
|
|
match op {
|
|
|
|
BinaryOp::ArithOp(b) => {
|
|
|
|
let panic_arith = ConstEvalError::Panic(
|
|
|
|
"attempt to run invalid arithmetic operation".to_string(),
|
|
|
|
);
|
|
|
|
let r = match b {
|
|
|
|
ArithOp::Add => v1.checked_add(v2).ok_or_else(|| panic_arith.clone())?,
|
|
|
|
ArithOp::Mul => v1.checked_mul(v2).ok_or_else(|| panic_arith.clone())?,
|
|
|
|
ArithOp::Sub => v1.checked_sub(v2).ok_or_else(|| panic_arith.clone())?,
|
|
|
|
ArithOp::Div => v1.checked_div(v2).ok_or_else(|| panic_arith.clone())?,
|
|
|
|
ArithOp::Rem => v1.checked_rem(v2).ok_or_else(|| panic_arith.clone())?,
|
|
|
|
ArithOp::Shl => v1
|
|
|
|
.checked_shl(v2.try_into().map_err(|_| panic_arith.clone())?)
|
|
|
|
.ok_or_else(|| panic_arith.clone())?,
|
|
|
|
ArithOp::Shr => v1
|
|
|
|
.checked_shr(v2.try_into().map_err(|_| panic_arith.clone())?)
|
|
|
|
.ok_or_else(|| panic_arith.clone())?,
|
|
|
|
ArithOp::BitXor => v1 ^ v2,
|
|
|
|
ArithOp::BitOr => v1 | v2,
|
|
|
|
ArithOp::BitAnd => v1 & v2,
|
|
|
|
};
|
|
|
|
if let TyKind::Scalar(s) = ty.kind(Interner) {
|
|
|
|
if !is_valid(s, r) {
|
|
|
|
return Err(panic_arith);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(ComputedExpr::Literal(Literal::Int(r, None)))
|
|
|
|
}
|
|
|
|
BinaryOp::LogicOp(_) => Err(ConstEvalError::TypeError),
|
|
|
|
_ => return Err(ConstEvalError::NotSupported("bin op on this operators")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Expr::Block { statements, tail, .. } => {
|
|
|
|
for statement in &**statements {
|
|
|
|
match statement {
|
|
|
|
&hir_def::expr::Statement::Let { pat, initializer, .. } => {
|
|
|
|
let pat = &ctx.pats[pat];
|
|
|
|
let name = match pat {
|
|
|
|
Pat::Bind { name, subpat, .. } if subpat.is_none() => name.clone(),
|
|
|
|
_ => {
|
|
|
|
return Err(ConstEvalError::NotSupported("complex patterns in let"))
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let value = match initializer {
|
|
|
|
Some(x) => eval_const(&ctx.exprs[x], ctx.clone())?,
|
|
|
|
None => continue,
|
|
|
|
};
|
|
|
|
ctx.local_data.insert(name, value);
|
|
|
|
}
|
|
|
|
&hir_def::expr::Statement::Expr { .. } => {
|
|
|
|
return Err(ConstEvalError::NotSupported("this kind of statement"))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let tail_expr = match tail {
|
|
|
|
&Some(x) => &ctx.exprs[x],
|
|
|
|
None => return Ok(ComputedExpr::Tuple(Box::new([]))),
|
|
|
|
};
|
|
|
|
eval_const(tail_expr, ctx)
|
|
|
|
}
|
|
|
|
Expr::Path(p) => {
|
|
|
|
let name = p.mod_path().as_ident().ok_or(ConstEvalError::NotSupported("big paths"))?;
|
|
|
|
let r = ctx
|
|
|
|
.local_data
|
|
|
|
.get(name)
|
|
|
|
.ok_or(ConstEvalError::NotSupported("Non local name resolution"))?;
|
|
|
|
Ok(r.clone())
|
|
|
|
}
|
|
|
|
_ => Err(ConstEvalError::NotSupported("This kind of expression")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-16 01:51:18 +00:00
|
|
|
// FIXME: support more than just evaluating literals
|
|
|
|
pub fn eval_usize(expr: &Expr) -> Option<u64> {
|
|
|
|
match expr {
|
2021-06-17 15:37:14 +00:00
|
|
|
Expr::Literal(Literal::Uint(v, None | Some(BuiltinUint::Usize))) => (*v).try_into().ok(),
|
2021-05-16 01:51:18 +00:00
|
|
|
_ => None,
|
2021-05-14 07:59:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Interns a possibly-unknown target usize
|
|
|
|
pub fn usize_const(value: Option<u64>) -> Const {
|
|
|
|
ConstData {
|
2021-12-19 16:58:39 +00:00
|
|
|
ty: TyKind::Scalar(chalk_ir::Scalar::Uint(chalk_ir::UintTy::Usize)).intern(Interner),
|
2021-05-14 07:59:30 +00:00
|
|
|
value: ConstValue::Concrete(chalk_ir::ConcreteConst {
|
2021-06-13 03:59:36 +00:00
|
|
|
interned: value.map(ConstScalar::Usize).unwrap_or(ConstScalar::Unknown),
|
2021-05-14 07:59:30 +00:00
|
|
|
}),
|
|
|
|
}
|
2021-12-19 16:58:39 +00:00
|
|
|
.intern(Interner)
|
2021-05-14 07:59:30 +00:00
|
|
|
}
|