travel fn path for compilation
This commit is contained in:
@@ -1,15 +1,15 @@
|
||||
use std::{collections::HashMap, ops::Deref};
|
||||
use std::collections::HashMap;
|
||||
|
||||
use crate::ir::AddrID;
|
||||
use crate::ir::Symbol;
|
||||
|
||||
pub fn create_program<I: Instr>(
|
||||
fns: Vec<(Vec<I>, AddrID)>,
|
||||
ro_data: Vec<(Vec<u8>, AddrID)>,
|
||||
start: Option<AddrID>,
|
||||
fns: Vec<(Vec<I>, Symbol)>,
|
||||
ro_data: Vec<(Vec<u8>, Symbol)>,
|
||||
start: Option<Symbol>,
|
||||
) -> (Vec<u8>, Option<Addr>) {
|
||||
let mut data = Vec::new();
|
||||
let mut sym_table = SymTable::new(fns.len() + ro_data.len());
|
||||
let mut missing = HashMap::<AddrID, Vec<(Addr, I)>>::new();
|
||||
let mut missing = HashMap::<Symbol, Vec<(Addr, I)>>::new();
|
||||
for (val, id) in ro_data {
|
||||
sym_table.insert(id, Addr(data.len() as u64));
|
||||
data.extend(val);
|
||||
@@ -45,7 +45,7 @@ pub fn create_program<I: Instr>(
|
||||
|
||||
pub trait Instr {
|
||||
fn push(&self, data: &mut Vec<u8>, syms: &SymTable, pos: Addr, missing: bool)
|
||||
-> Option<AddrID>;
|
||||
-> Option<Symbol>;
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Copy, PartialEq)]
|
||||
@@ -57,75 +57,16 @@ impl Addr {
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Copy, Hash, PartialEq, Eq)]
|
||||
pub struct Symbol(usize);
|
||||
/// intentionally does not have copy or clone;
|
||||
/// this should only be consumed once
|
||||
pub struct WritableSymbol(Symbol);
|
||||
|
||||
impl Deref for WritableSymbol {
|
||||
type Target = Symbol;
|
||||
fn deref(&self) -> &Self::Target {
|
||||
&self.0
|
||||
}
|
||||
}
|
||||
|
||||
pub struct SymMap<I> {
|
||||
i: usize,
|
||||
ro_data: Vec<(Vec<u8>, Symbol)>,
|
||||
functions: Vec<(Vec<I>, Symbol)>,
|
||||
}
|
||||
|
||||
impl<I> SymMap<I> {
|
||||
pub fn new() -> Self {
|
||||
Self {
|
||||
i: 0,
|
||||
ro_data: Vec::new(),
|
||||
functions: Vec::new(),
|
||||
}
|
||||
}
|
||||
pub fn push_ro_data(&mut self, data: Vec<u8>) -> Symbol {
|
||||
let sym = self.reserve();
|
||||
self.write_ro_data(sym, data.into())
|
||||
}
|
||||
pub fn push_ro_data_size(&mut self, data: Vec<u8>) -> (Symbol, usize) {
|
||||
let sym = self.reserve();
|
||||
let len = data.len();
|
||||
(self.write_ro_data(sym, data), len)
|
||||
}
|
||||
pub fn push_fn(&mut self, instructions: Vec<I>) -> Symbol {
|
||||
let sym = self.reserve();
|
||||
self.write_fn(sym, instructions)
|
||||
}
|
||||
pub fn write_ro_data(&mut self, sym: WritableSymbol, data: Vec<u8>) -> Symbol {
|
||||
let data = data.into();
|
||||
self.ro_data.push((data, *sym));
|
||||
*sym
|
||||
}
|
||||
pub fn write_fn(&mut self, sym: WritableSymbol, instructions: Vec<I>) -> Symbol {
|
||||
self.functions.push((instructions, *sym));
|
||||
*sym
|
||||
}
|
||||
pub fn reserve(&mut self) -> WritableSymbol {
|
||||
let val = self.i;
|
||||
self.i += 1;
|
||||
WritableSymbol(Symbol(val))
|
||||
}
|
||||
pub fn len(&self) -> usize {
|
||||
self.functions.len() + self.ro_data.len()
|
||||
}
|
||||
}
|
||||
|
||||
pub struct SymTable(Vec<Addr>);
|
||||
impl SymTable {
|
||||
pub fn new(len: usize) -> Self {
|
||||
Self(vec![Addr::NONE; len])
|
||||
}
|
||||
pub fn insert(&mut self, sym: AddrID, addr: Addr) {
|
||||
self.0[sym.0] = addr;
|
||||
pub fn insert(&mut self, sym: Symbol, addr: Addr) {
|
||||
self.0[*sym] = addr;
|
||||
}
|
||||
pub fn get(&self, sym: AddrID) -> Option<Addr> {
|
||||
match self.0[sym.0] {
|
||||
pub fn get(&self, sym: Symbol) -> Option<Addr> {
|
||||
match self.0[*sym] {
|
||||
Addr::NONE => None,
|
||||
addr => Some(addr),
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user