forked from M-Labs/nac3
David Mak
77de24ef74
There have been multiple instances where I had the need to iterate over type variables, only to discover that the traversal order is arbitrary. This commit fixes that by adding SortedMapping, which utilizes BTreeMap internally to guarantee a traversal order. All instances of VarMap are now refactored to use this to ensure that type variables are iterated in the order of its variable ID, which should be monotonically incremented by the unifier.
144 lines
4.5 KiB
Rust
144 lines
4.5 KiB
Rust
use std::{
|
|
borrow::BorrowMut,
|
|
collections::{HashMap, HashSet},
|
|
fmt::Debug,
|
|
iter::FromIterator,
|
|
sync::Arc,
|
|
};
|
|
|
|
use super::codegen::CodeGenContext;
|
|
use super::typecheck::type_inferencer::PrimitiveStore;
|
|
use super::typecheck::typedef::{FunSignature, FuncArg, SharedUnifier, Type, TypeEnum, Unifier, VarMap};
|
|
use crate::{
|
|
codegen::CodeGenerator,
|
|
symbol_resolver::{SymbolResolver, ValueEnum},
|
|
typecheck::{type_inferencer::CodeLocation, typedef::CallId},
|
|
};
|
|
use inkwell::values::BasicValueEnum;
|
|
use itertools::{izip, Itertools};
|
|
use nac3parser::ast::{self, Location, Stmt, StrRef};
|
|
use parking_lot::RwLock;
|
|
|
|
#[derive(PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Hash, Debug)]
|
|
pub struct DefinitionId(pub usize);
|
|
|
|
pub mod builtins;
|
|
pub mod composer;
|
|
pub mod helper;
|
|
pub mod numpy;
|
|
pub mod type_annotation;
|
|
use composer::*;
|
|
use type_annotation::*;
|
|
#[cfg(test)]
|
|
mod test;
|
|
|
|
type GenCallCallback = Box<
|
|
dyn for<'ctx, 'a> Fn(
|
|
&mut CodeGenContext<'ctx, 'a>,
|
|
Option<(Type, ValueEnum<'ctx>)>,
|
|
(&FunSignature, DefinitionId),
|
|
Vec<(Option<StrRef>, ValueEnum<'ctx>)>,
|
|
&mut dyn CodeGenerator,
|
|
) -> Result<Option<BasicValueEnum<'ctx>>, String>
|
|
+ Send
|
|
+ Sync,
|
|
>;
|
|
|
|
pub struct GenCall {
|
|
fp: GenCallCallback,
|
|
}
|
|
|
|
impl GenCall {
|
|
#[must_use]
|
|
pub fn new(fp: GenCallCallback) -> GenCall {
|
|
GenCall { fp }
|
|
}
|
|
|
|
pub fn run<'ctx>(
|
|
&self,
|
|
ctx: &mut CodeGenContext<'ctx, '_>,
|
|
obj: Option<(Type, ValueEnum<'ctx>)>,
|
|
fun: (&FunSignature, DefinitionId),
|
|
args: Vec<(Option<StrRef>, ValueEnum<'ctx>)>,
|
|
generator: &mut dyn CodeGenerator,
|
|
) -> Result<Option<BasicValueEnum<'ctx>>, String> {
|
|
(self.fp)(ctx, obj, fun, args, generator)
|
|
}
|
|
}
|
|
|
|
impl Debug for GenCall {
|
|
fn fmt(&self, _: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug)]
|
|
pub struct FunInstance {
|
|
pub body: Arc<Vec<Stmt<Option<Type>>>>,
|
|
pub calls: Arc<HashMap<CodeLocation, CallId>>,
|
|
pub subst: VarMap,
|
|
pub unifier_id: usize,
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub enum TopLevelDef {
|
|
Class {
|
|
/// Name for error messages and symbols.
|
|
name: StrRef,
|
|
/// Object ID used for [TypeEnum].
|
|
object_id: DefinitionId,
|
|
/// type variables bounded to the class.
|
|
type_vars: Vec<Type>,
|
|
/// Class fields.
|
|
///
|
|
/// Name and type is mutable.
|
|
fields: Vec<(StrRef, Type, bool)>,
|
|
/// Class methods, pointing to the corresponding function definition.
|
|
methods: Vec<(StrRef, Type, DefinitionId)>,
|
|
/// Ancestor classes, including itself.
|
|
ancestors: Vec<TypeAnnotation>,
|
|
/// Symbol resolver of the module defined the class; [None] if it is built-in type.
|
|
resolver: Option<Arc<dyn SymbolResolver + Send + Sync>>,
|
|
/// Constructor type.
|
|
constructor: Option<Type>,
|
|
/// Definition location.
|
|
loc: Option<Location>,
|
|
},
|
|
Function {
|
|
/// Prefix for symbol, should be unique globally.
|
|
name: String,
|
|
/// Simple name, the same as in method/function definition.
|
|
simple_name: StrRef,
|
|
/// Function signature.
|
|
signature: Type,
|
|
/// Instantiated type variable IDs.
|
|
var_id: Vec<u32>,
|
|
/// Function instance to symbol mapping
|
|
///
|
|
/// * Key: String representation of type variable values, sorted by variable ID in ascending
|
|
/// order, including type variables associated with the class.
|
|
/// * Value: Function symbol name.
|
|
instance_to_symbol: HashMap<String, String>,
|
|
/// Function instances to annotated AST mapping
|
|
///
|
|
/// * Key: String representation of type variable values, sorted by variable ID in ascending
|
|
/// order, including type variables associated with the class. Excluding rigid type
|
|
/// variables.
|
|
///
|
|
/// Rigid type variables that would be substituted when the function is instantiated.
|
|
instance_to_stmt: HashMap<String, FunInstance>,
|
|
/// Symbol resolver of the module defined the class.
|
|
resolver: Option<Arc<dyn SymbolResolver + Send + Sync>>,
|
|
/// Custom code generation callback.
|
|
codegen_callback: Option<Arc<GenCall>>,
|
|
/// Definition location.
|
|
loc: Option<Location>,
|
|
},
|
|
}
|
|
|
|
pub struct TopLevelContext {
|
|
pub definitions: Arc<RwLock<Vec<Arc<RwLock<TopLevelDef>>>>>,
|
|
pub unifiers: Arc<RwLock<Vec<(SharedUnifier, PrimitiveStore)>>>,
|
|
pub personality_symbol: Option<String>,
|
|
}
|