1
0

202 lines
6.6 KiB
Rust
Raw Normal View History

2024-11-15 19:18:00 -05:00
//! Runtime values used by the VM.
2024-11-15 21:42:27 -05:00
mod abstract_value;
mod concrete_value;
mod range_value;
2024-11-15 19:18:00 -05:00
2024-11-15 21:42:27 -05:00
pub use abstract_value::AbstractValue;
pub use concrete_value::{ConcreteValue, DustString};
2024-11-15 21:42:27 -05:00
pub use range_value::RangeValue;
2024-12-14 08:49:02 -05:00
use serde::{Deserialize, Serialize};
2024-11-15 19:18:00 -05:00
use std::fmt::{self, Debug, Display, Formatter};
2024-12-14 16:17:02 -05:00
use crate::{Type, Vm};
2024-11-15 19:18:00 -05:00
2024-12-14 08:49:02 -05:00
#[derive(Clone, Debug, PartialEq, PartialOrd, Serialize, Deserialize)]
2024-11-17 20:32:53 -05:00
pub enum Value {
2024-12-14 08:49:02 -05:00
#[serde(skip)]
2024-11-16 01:29:21 -05:00
Abstract(AbstractValue),
Concrete(ConcreteValue),
2024-11-15 19:18:00 -05:00
}
2024-11-17 20:32:53 -05:00
impl Value {
2024-12-14 16:17:02 -05:00
pub fn into_concrete_owned<'a>(self, vm: &'a Vm<'a>) -> ConcreteValue {
2024-11-15 21:42:27 -05:00
match self {
2024-11-17 20:32:53 -05:00
Value::Abstract(abstract_value) => abstract_value.to_concrete_owned(vm),
Value::Concrete(concrete_value) => concrete_value,
2024-11-15 19:18:00 -05:00
}
}
2024-12-04 05:10:53 -05:00
pub fn as_boolean(&self) -> Option<&bool> {
if let Value::Concrete(ConcreteValue::Boolean(value)) = self {
Some(value)
} else {
None
}
}
2024-12-14 08:49:02 -05:00
pub fn r#type(&self) -> Type {
match self {
2024-12-14 08:49:02 -05:00
Value::Abstract(abstract_value) => abstract_value.r#type(),
Value::Concrete(concrete_value) => concrete_value.r#type(),
}
}
2024-12-14 08:49:02 -05:00
pub fn add(&self, other: &Value) -> Result<Value, ValueError> {
match (self, other) {
2024-12-14 08:49:02 -05:00
(Value::Concrete(left), Value::Concrete(right)) => left.add(right).map(Value::Concrete),
_ => Err(ValueError::CannotAdd(self.to_owned(), other.to_owned())),
}
}
2024-12-14 08:49:02 -05:00
pub fn subtract(&self, other: &Value) -> Result<Value, ValueError> {
match (self, other) {
2024-12-14 08:49:02 -05:00
(Value::Concrete(left), Value::Concrete(right)) => {
2024-12-04 00:04:56 -05:00
left.subtract(right).map(Value::Concrete)
}
_ => Err(ValueError::CannotSubtract(
self.to_owned(),
other.to_owned(),
)),
}
}
2024-12-14 08:49:02 -05:00
pub fn multiply(&self, other: &Value) -> Result<Value, ValueError> {
match (self, other) {
2024-12-14 08:49:02 -05:00
(Value::Concrete(left), Value::Concrete(right)) => {
2024-12-04 00:04:56 -05:00
left.multiply(right).map(Value::Concrete)
}
_ => Err(ValueError::CannotMultiply(
self.to_owned(),
other.to_owned(),
)),
}
}
2024-12-14 08:49:02 -05:00
pub fn divide(&self, other: &Value) -> Result<Value, ValueError> {
match (self, other) {
2024-12-14 08:49:02 -05:00
(Value::Concrete(left), Value::Concrete(right)) => {
2024-12-04 00:04:56 -05:00
left.divide(right).map(Value::Concrete)
}
_ => Err(ValueError::CannotDivide(self.to_owned(), other.to_owned())),
}
}
2024-12-14 08:49:02 -05:00
pub fn modulo(&self, other: &Value) -> Result<Value, ValueError> {
match (self, other) {
2024-12-14 08:49:02 -05:00
(Value::Concrete(left), Value::Concrete(right)) => {
2024-12-04 00:04:56 -05:00
left.modulo(right).map(Value::Concrete)
}
_ => Err(ValueError::CannotModulo(self.to_owned(), other.to_owned())),
}
}
pub fn negate(&self) -> Result<Value, ValueError> {
match self {
2024-12-14 08:49:02 -05:00
Value::Concrete(concrete_value) => concrete_value.negate().map(Value::Concrete),
_ => Err(ValueError::CannotNegate(self.to_owned())),
}
}
pub fn not(&self) -> Result<Value, ValueError> {
match self {
2024-12-14 08:49:02 -05:00
Value::Concrete(concrete_value) => concrete_value.not().map(Value::Concrete),
_ => Err(ValueError::CannotNot(self.to_owned())),
}
}
2024-12-14 08:49:02 -05:00
pub fn equal(&self, other: &Value) -> Result<Value, ValueError> {
match (self, other) {
2024-12-14 08:49:02 -05:00
(Value::Concrete(left), Value::Concrete(right)) => {
2024-12-04 00:04:56 -05:00
left.equal(right).map(Value::Concrete)
}
_ => Err(ValueError::CannotCompare(self.to_owned(), other.to_owned())),
}
}
2024-12-14 08:49:02 -05:00
pub fn less(&self, other: &Value) -> Result<Value, ValueError> {
match (self, other) {
2024-12-14 08:49:02 -05:00
(Value::Concrete(left), Value::Concrete(right)) => {
2024-12-04 00:04:56 -05:00
left.less_than(right).map(Value::Concrete)
}
_ => Err(ValueError::CannotCompare(self.to_owned(), other.to_owned())),
}
}
2024-12-14 08:49:02 -05:00
pub fn less_equal(&self, other: &Value) -> Result<Value, ValueError> {
match (self, other) {
2024-12-14 08:49:02 -05:00
(Value::Concrete(left), Value::Concrete(right)) => {
2024-12-04 00:04:56 -05:00
left.less_than_or_equal(right).map(Value::Concrete)
}
_ => Err(ValueError::CannotCompare(self.to_owned(), other.to_owned())),
}
}
2024-12-14 08:49:02 -05:00
2024-12-14 16:17:02 -05:00
pub fn display(&self, vm: &Vm) -> DustString {
2024-12-14 08:49:02 -05:00
match self {
2024-12-14 16:17:02 -05:00
Value::Abstract(abstract_value) => abstract_value.display(vm),
Value::Concrete(concrete_value) => concrete_value.display(),
2024-12-14 08:49:02 -05:00
}
}
}
2024-12-14 08:49:02 -05:00
impl Display for Value {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
match self {
2024-12-14 08:49:02 -05:00
Value::Abstract(abstract_value) => write!(f, "{}", abstract_value),
Value::Concrete(concrete_value) => write!(f, "{}", concrete_value),
}
}
2024-11-15 19:18:00 -05:00
}
#[derive(Clone, Debug, PartialEq)]
pub enum ValueError {
CannotAdd(Value, Value),
CannotAnd(Value, Value),
CannotCompare(Value, Value),
CannotDivide(Value, Value),
CannotModulo(Value, Value),
CannotMultiply(Value, Value),
CannotNegate(Value),
CannotNot(Value),
CannotSubtract(Value, Value),
CannotOr(Value, Value),
2024-11-15 19:18:00 -05:00
}
impl Display for ValueError {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
match self {
ValueError::CannotAdd(left, right) => {
write!(f, "Cannot add {left} and {right}")
}
ValueError::CannotAnd(left, right) => {
write!(f, "Cannot use logical AND operation on {left} and {right}")
}
ValueError::CannotCompare(left, right) => {
write!(f, "Cannot compare {left} and {right}")
}
ValueError::CannotDivide(left, right) => {
write!(f, "Cannot divide {left} by {right}")
}
ValueError::CannotModulo(left, right) => {
write!(f, "Cannot use modulo operation on {left} and {right}")
}
ValueError::CannotMultiply(left, right) => {
write!(f, "Cannot multiply {left} by {right}")
}
ValueError::CannotNegate(value) => {
write!(f, "Cannot negate {value}")
}
ValueError::CannotNot(value) => {
write!(f, "Cannot use logical NOT operation on {value}")
}
ValueError::CannotSubtract(left, right) => {
write!(f, "Cannot subtract {right} from {left}")
}
ValueError::CannotOr(left, right) => {
write!(f, "Cannot use logical OR operation on {left} and {right}")
}
}
}
}