use cervine::Cow;
use super::*;
pub fn add(vm: &mut VM, (lhs, rhs): (RantValue, RantValue)) -> RantStdResult {
vm.cur_frame_mut().write(lhs + rhs);
Ok(())
}
pub fn clamp(vm: &mut VM, (value, a, b): (RantValue, RantValue, RantValue)) -> RantStdResult {
vm.cur_frame_mut().write(util::clamp(value, a, b));
Ok(())
}
pub fn mul(vm: &mut VM, (lhs, rhs): (RantValue, RantValue)) -> RantStdResult {
vm.cur_frame_mut().write(lhs * rhs);
Ok(())
}
pub fn mul_add(vm: &mut VM, (lhs, mhs, rhs): (RantValue, RantValue, RantValue)) -> RantStdResult {
vm.cur_frame_mut().write(lhs * mhs + rhs);
Ok(())
}
pub fn sub(vm: &mut VM, (lhs, rhs): (RantValue, RantValue)) -> RantStdResult {
vm.cur_frame_mut().write(lhs - rhs);
Ok(())
}
pub fn div(vm: &mut VM, (lhs, rhs): (RantValue, RantValue)) -> RantStdResult {
vm.cur_frame_mut().write((lhs / rhs).into_runtime_result()?);
Ok(())
}
pub fn mod_(vm: &mut VM, (lhs, rhs): (RantValue, RantValue)) -> RantStdResult {
vm.cur_frame_mut().write((lhs % rhs).into_runtime_result()?);
Ok(())
}
pub fn neg(vm: &mut VM, val: RantValue) -> RantStdResult {
vm.cur_frame_mut().write(-val);
Ok(())
}
pub fn recip(vm: &mut VM, val: RantValue) -> RantStdResult {
vm.cur_frame_mut().write((RantValue::Float(1.0) / val).into_runtime_result()?);
Ok(())
}
pub fn floor(vm: &mut VM, val: RantValue) -> RantStdResult {
let val_result = match val {
RantValue::Float(f) => RantValue::Float(f.floor()),
RantValue::Int(i) => RantValue::Int(i),
other => runtime_error!(RuntimeErrorType::ArgumentError, "cannot use floor function on '{}' value")
};
vm.cur_frame_mut().write(val_result);
Ok(())
}
pub fn ceil(vm: &mut VM, val: RantValue) -> RantStdResult {
let val_result = match val {
RantValue::Float(f) => RantValue::Float(f.ceil()),
RantValue::Int(i) => RantValue::Int(i),
other => runtime_error!(RuntimeErrorType::ArgumentError, "cannot use ceil function on '{}' value")
};
vm.cur_frame_mut().write(val_result);
Ok(())
}
pub fn frac(vm: &mut VM, val: RantValue) -> RantStdResult {
let val_result = match val {
RantValue::Float(f) => RantValue::Float(f.fract()),
other => runtime_error!(RuntimeErrorType::ArgumentError, "cannot use frac function on '{}' value")
};
vm.cur_frame_mut().write(val_result);
Ok(())
}
pub fn abs(vm: &mut VM, num: RantValue) -> RantStdResult {
vm.cur_frame_mut().write(num.abs().into_runtime_result()?);
Ok(())
}
pub fn pow(vm: &mut VM, (lhs, rhs): (RantValue, RantValue)) -> RantStdResult {
vm.cur_frame_mut().write(lhs.pow(rhs).into_runtime_result()?);
Ok(())
}
pub fn sin(vm: &mut VM, x: f64) -> RantStdResult {
vm.cur_frame_mut().write(x.sin());
Ok(())
}
pub fn asin(vm: &mut VM, x: f64) -> RantStdResult {
vm.cur_frame_mut().write(x.asin());
Ok(())
}
pub fn cos(vm: &mut VM, x: f64) -> RantStdResult {
vm.cur_frame_mut().write(x.cos());
Ok(())
}
pub fn acos(vm: &mut VM, x: f64) -> RantStdResult {
vm.cur_frame_mut().write(x.acos());
Ok(())
}
pub fn tan(vm: &mut VM, x: f64) -> RantStdResult {
vm.cur_frame_mut().write(x.tan());
Ok(())
}
pub fn atan(vm: &mut VM, x: f64) -> RantStdResult {
vm.cur_frame_mut().write(x.atan());
Ok(())
}
pub fn atan2(vm: &mut VM, (y, x): (f64, f64)) -> RantStdResult {
vm.cur_frame_mut().write(y.atan2(x));
Ok(())
}
pub fn sqrt(vm: &mut VM, num: RantNumber) -> RantStdResult {
let result = match num {
RantNumber::Int(i) => (i as f64).sqrt(),
RantNumber::Float(f) => f.sqrt(),
};
vm.cur_frame_mut().write(result);
Ok(())
}
pub fn min(vm: &mut VM, values: RequiredVarArgs<RantValue>) -> RantStdResult {
if values.is_empty() {
return Ok(())
}
let iter = values.iter();
let mut min = None;
for val in iter {
let flat_val = match val {
RantValue::List(list_ref) => {
Some(Cow::Owned(util::min_rant_value(list_ref.borrow().iter()).cloned().unwrap_or_default()))
},
RantValue::Tuple(tuple_ref) => {
Some(Cow::Owned(util::min_rant_value(tuple_ref.iter()).cloned().unwrap_or_default()))
}
other => Some(Cow::Borrowed(other)),
};
if min.is_none() || flat_val < min {
min = flat_val;
}
}
vm.cur_frame_mut().write(match min.unwrap() {
Cow::Owned(val) => val,
Cow::Borrowed(val) => val.clone(),
});
Ok(())
}
pub fn max(vm: &mut VM, values: RequiredVarArgs<RantValue>) -> RantStdResult {
if values.is_empty() {
return Ok(())
}
let iter = values.iter();
let mut max = None;
for val in iter {
let flat_val = match val {
RantValue::List(list_ref) => {
Some(Cow::Owned(util::max_rant_value(list_ref.borrow().iter()).cloned().unwrap_or_default()))
},
RantValue::Tuple(tuple_ref) => {
Some(Cow::Owned(util::max_rant_value(tuple_ref.iter()).cloned().unwrap_or_default()))
}
other => Some(Cow::Borrowed(other)),
};
if max.is_none() || flat_val >= max {
max = flat_val;
}
}
vm.cur_frame_mut().write(match max.unwrap() {
Cow::Owned(val) => val,
Cow::Borrowed(val) => val.clone(),
});
Ok(())
}