aboutsummaryrefslogtreecommitdiffstats
path: root/src/vm.rs
diff options
context:
space:
mode:
authorEdvard Thörnros <edvard.thornros@gmail.com>2021-01-20 21:26:14 +0100
committerEdvard Thörnros <edvard.thornros@gmail.com>2021-01-20 21:26:14 +0100
commitd13695d2452e0468168240800a4cde88f220b646 (patch)
tree35a8ae5e9bf656bc4a3435bab1afc367757aa8a7 /src/vm.rs
parent85411e444826fc239fb268da03572993d9c9fd83 (diff)
downloadsylt-d13695d2452e0468168240800a4cde88f220b646.tar.gz
Typecheck upvalues
Diffstat (limited to 'src/vm.rs')
-rw-r--r--src/vm.rs18
1 files changed, 16 insertions, 2 deletions
diff --git a/src/vm.rs b/src/vm.rs
index 7b48b4e..abe7874 100644
--- a/src/vm.rs
+++ b/src/vm.rs
@@ -152,7 +152,7 @@ pub enum Op {
#[derive(Debug)]
pub struct Block {
pub ty: Type,
- pub ups: Vec<Type>,
+ pub ups: Vec<(usize, bool, Type)>,
pub name: String,
pub file: PathBuf,
@@ -521,7 +521,7 @@ impl VM {
Op::Call(num_args) => {
let new_base = self.stack.len() - 1 - num_args;
match &self.stack[new_base] {
- Value::Function(_, block) => {
+ Value::Function(ups, block) => {
let args = block.args();
if args.len() != num_args {
error!(self,
@@ -613,6 +613,19 @@ impl VM {
Op::Jmp(_line) => {}
+ Op::ReadUpvalue(slot) => {
+ self.stack.push(self.frame().block.ups[slot].2.as_value());
+ }
+
+ Op::AssignUpvalue(slot) => {
+ let var = self.frame().block.ups[slot].2.clone();
+ let up = self.stack.pop().unwrap().as_type();
+ if var != up {
+ error!(self, ErrorKind::TypeError(op, vec![var, up]),
+ "Incorrect type for upvalue.".to_string());
+ }
+ }
+
Op::Return => {
let a = self.stack.pop().unwrap();
let ret = self.frame().block.ret();
@@ -738,6 +751,7 @@ impl VM {
let mut errors = Vec::new();
for block in blocks.iter() {
+ let ups: Vec<_> = block.ups.iter().map(|x| x.2.as_value()).collect();
errors.append(&mut self.typecheck_block(Rc::clone(block)));
}