rudus/src/compiler.rs

186 lines
5.7 KiB
Rust
Raw Normal View History

use crate::parser::Ast;
use crate::spans::Spanned;
use crate::value::*;
use chumsky::prelude::SimpleSpan;
use num_derive::{FromPrimitive, ToPrimitive};
use num_traits::FromPrimitive;
#[derive(Copy, Clone, Debug, PartialEq, Eq, FromPrimitive, ToPrimitive)]
2024-12-16 04:28:57 +00:00
pub enum Op {
Return,
Constant,
2024-12-16 04:28:57 +00:00
Jump,
JumpIfFalse,
}
impl std::fmt::Display for Op {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
use Op::*;
match self {
Return => write!(f, "return"),
Constant => write!(f, "constant"),
Jump => write!(f, "jump"),
JumpIfFalse => write!(f, "jump_if_false"),
}
}
}
#[derive(Clone, Debug, PartialEq)]
2024-12-16 04:28:57 +00:00
pub struct Local {
name: &'static str,
depth: u8,
}
#[derive(Clone, Debug, PartialEq)]
pub struct Chunk<'a> {
pub locals: Vec<Local>,
scope_depth: usize,
local_count: usize,
pub constants: Vec<Value>,
pub bytecode: Vec<u8>,
pub spans: Vec<SimpleSpan>,
pub strings: Vec<&'static str>,
pub keywords: Vec<&'static str>,
pub nodes: Vec<&'a Ast>,
pub ast: &'a Ast,
pub span: SimpleSpan,
pub src: &'static str,
pub name: &'static str,
}
impl<'a> Chunk<'a> {
2024-12-16 04:28:57 +00:00
pub fn new(ast: &'a Spanned<Ast>, name: &'static str, src: &'static str) -> Chunk<'a> {
Chunk {
locals: vec![],
scope_depth: 0,
local_count: 0,
constants: vec![],
bytecode: vec![],
spans: vec![],
strings: vec![],
keywords: vec![
"nil", "bool", "number", "keyword", "string", "tuple", "list", "",
],
nodes: vec![],
ast: &ast.0,
span: ast.1,
src,
name,
}
}
pub fn kw_from(&self, kw: &str) -> Option<Value> {
self.keywords
.iter()
.position(|s| *s == kw)
.map(Value::Keyword)
}
pub fn visit(&mut self, node: &'a Spanned<Ast>) {
let root_node = self.ast;
let root_span = self.span;
let (ast, span) = node;
self.ast = ast;
self.span = *span;
self.compile();
self.ast = root_node;
self.span = root_span;
}
2024-12-16 04:28:57 +00:00
fn emit_constant(&mut self, val: Value) {
let constant_index = self.constants.len();
if constant_index > u8::MAX as usize {
panic!(
"internal Ludus compiler error: too many constants in chunk:{}:: {}",
self.span, self.ast
)
}
self.constants.push(val);
self.bytecode.push(Op::Constant as u8);
self.spans.push(self.span);
self.bytecode.push(constant_index as u8);
self.spans.push(self.span);
}
fn emit_op(&mut self, op: Op) {
self.bytecode.push(op as u8);
self.spans.push(self.span);
}
2024-12-16 04:28:57 +00:00
pub fn compile(&mut self) {
use Ast::*;
match self.ast {
2024-12-16 04:28:57 +00:00
Nil => self.emit_constant(Value::Nil),
Number(n) => self.emit_constant(Value::Number(*n)),
Boolean(b) => self.emit_constant(if *b { Value::True } else { Value::False }),
String(s) => {
let str_index = self.strings.len();
self.strings.push(s);
2024-12-16 04:28:57 +00:00
self.emit_constant(Value::Interned(str_index));
}
2024-12-16 04:28:57 +00:00
Keyword(s) => {
let existing_kw = self.keywords.iter().position(|kw| kw == s);
let kw_index = match existing_kw {
Some(index) => index,
None => self.keywords.len(),
};
self.keywords.push(s);
self.emit_constant(Value::Keyword(kw_index));
}
Block(lines) => {
self.scope_depth += 1;
for expr in lines {
self.visit(expr);
}
self.emit_op(Op::Return);
2024-12-16 04:28:57 +00:00
self.scope_depth -= 1;
}
If(cond, then, r#else) => {
self.visit(cond);
let jif_idx = self.bytecode.len();
self.emit_op(Op::JumpIfFalse);
self.bytecode.push(0xff);
self.visit(then);
let jump_idx = self.bytecode.len();
self.emit_op(Op::Jump);
self.bytecode.push(0xff);
self.visit(r#else);
let end_idx = self.bytecode.len();
let jif_offset = jump_idx - jif_idx;
let jump_offset = end_idx - jump_idx;
self.bytecode[jif_idx + 1] = jif_offset as u8;
self.bytecode[jump_idx + 1] = jump_offset as u8;
}
2024-12-16 04:28:57 +00:00
// Let(patt, expr) => {
// self.visit(expr);
// self.visit(patt);
// }
// WordPattern(name) => {}
// PlaceholderPattern => {}
_ => todo!(),
}
}
2024-12-16 04:28:57 +00:00
pub fn disassemble(&self) {
println!("=== chunk: {} ===", self.name);
2024-12-16 04:28:57 +00:00
println!("IDX | CODE | INFO");
let mut codes = self.bytecode.iter().enumerate();
while let Some((i, byte)) = codes.next() {
2024-12-16 04:28:57 +00:00
let op = Op::from_u8(*byte).unwrap();
use Op::*;
match op {
2024-12-16 04:28:57 +00:00
Return => println!("{i:04}: {op}"),
Constant => {
let (_, next) = codes.next().unwrap();
let value = &self.constants[*next as usize].show(self);
println!("{i:04}: {:16} {next:04}: {value}", op.to_string());
}
Jump | JumpIfFalse => {
let (_, next) = codes.next().unwrap();
2024-12-16 04:28:57 +00:00
println!("{i:04}: {:16} {next:04}", op.to_string())
}
}
}
}
}