lisette-semantics 0.2.12

Little language inspired by Rust that compiles to Go
Documentation
pub(crate) mod addressability;
mod carry_mut;
pub(crate) mod expressions;
mod interface;
mod unify;
mod validation;

pub(crate) use unify::BuiltinBound;

use rustc_hash::{FxHashMap as HashMap, FxHashSet as HashSet};

use super::freeze::FreezeFolder;
use super::{FileContextKind, TaskState};
use crate::store::Store;
use syntax::ast::Expression;
use syntax::program::{File, FileImport};

impl TaskState<'_> {
    /// Infer types for `files` belonging to `module_id`.
    pub fn infer_module(&mut self, store: &Store, module_id: &str, files: Vec<File>) {
        let items_per_file: Vec<&[Expression]> = files.iter().map(|f| f.items.as_slice()).collect();
        self.check_const_cycles(store, &items_per_file);

        for file in files {
            self.infer_file(store, module_id, file);
        }
    }

    /// Extract a module's `.lis` files from the store.
    pub fn take_module_files(&mut self, store: &mut Store, module_id: &str) -> Vec<File> {
        self.with_module_cursor(module_id, |_this| {
            let module = store
                .get_module_mut(module_id)
                .expect("module must exist for inference");
            std::mem::take(&mut module.files).into_values().collect()
        })
    }

    fn infer_file(&mut self, store: &Store, module_id: &str, file: File) {
        let file_id = file.id;
        let imports = file.imports();

        self.with_file_context(
            store,
            module_id,
            file_id,
            &imports,
            FileContextKind::Standard,
            |this, store| {
                this.check_definition_module_collisions(store, &file.items, &imports);

                let inferred_items: Vec<_> = file
                    .items
                    .into_iter()
                    .map(|item| {
                        let type_var = this.new_type_var();
                        this.infer_expression(store, item, &type_var)
                    })
                    .collect();

                this.check_reference_sibling_aliasing(&inferred_items);

                let folder = FreezeFolder::new(&this.env);
                folder.freeze_facts(&mut this.facts);
                let frozen_items = FreezeFolder::new(&this.env).freeze_items(inferred_items);

                let typed_file = File {
                    id: file_id,
                    module_id: file.module_id,
                    name: file.name,
                    display_path: file.display_path,
                    source: file.source,
                    items: frozen_items,
                };

                this.typed_files.push((module_id.to_string(), typed_file));
            },
        );
    }

    fn check_definition_module_collisions(
        &mut self,
        store: &Store,
        items: &[Expression],
        imports: &[FileImport],
    ) {
        let alias_to_path: HashMap<String, String> = imports
            .iter()
            .filter_map(|imp| {
                imp.effective_alias(&store.go_package_names)
                    .map(|alias| (alias, imp.name.to_string()))
            })
            .collect();

        for item in items {
            let (definition_name, name_span) = match item {
                Expression::Function {
                    name, name_span, ..
                } => (name.as_str(), *name_span),
                Expression::Struct {
                    name, name_span, ..
                } => (name.as_str(), *name_span),
                Expression::Enum {
                    name, name_span, ..
                } => (name.as_str(), *name_span),
                Expression::ValueEnum {
                    name, name_span, ..
                } => (name.as_str(), *name_span),
                Expression::TypeAlias {
                    name, name_span, ..
                } => (name.as_str(), *name_span),
                Expression::Const {
                    identifier,
                    identifier_span,
                    ..
                } => (identifier.as_str(), *identifier_span),
                Expression::Interface {
                    name, name_span, ..
                } => (name.as_str(), *name_span),
                _ => continue,
            };

            if let Some(import_path) = alias_to_path.get(definition_name) {
                self.sink
                    .push(diagnostics::infer::definition_shadows_import(
                        definition_name,
                        import_path,
                        name_span,
                    ));
            }
        }
    }

    pub(crate) fn register_block_local_items(&mut self, store: &Store, items: &[Expression]) {
        for item in items {
            match item {
                Expression::Const { .. } => self.register_block_local_const(store, item),
                Expression::Function { .. } => self.register_block_local_fn(store, item),
                _ => {}
            }
        }
    }

    fn register_block_local_const(&mut self, store: &Store, item: &Expression) {
        let Expression::Const {
            identifier,
            identifier_span,
            annotation,
            expression,
            span,
            ..
        } = item
        else {
            return;
        };

        let qualified_name = self.qualify_name(identifier);
        let is_duplicate = self.scopes.lookup_const(identifier)
            || self.is_const_name(store, qualified_name.as_str());
        if is_duplicate && self.is_lis(store) {
            self.sink.push(diagnostics::infer::duplicate_definition(
                "constant",
                identifier,
                *identifier_span,
            ));
            return;
        }

        let before = self.sink.len();
        let const_ty = if let Some(annotation) = annotation {
            self.convert_to_type(store, annotation, span)
        } else {
            self.type_from_literal_expression(expression)
                .unwrap_or_else(|| self.new_type_var())
        };
        self.sink.truncate(before);

        let scope = self.scopes.current_mut();
        scope.values.insert(identifier.to_string(), const_ty);
        scope
            .consts
            .get_or_insert_with(HashSet::default)
            .insert(identifier.to_string());
    }

    fn register_block_local_fn(&mut self, store: &Store, item: &Expression) {
        let Expression::Function { name, span, .. } = item else {
            return;
        };

        let fn_sig = item.to_function_signature();

        let before = self.sink.len();
        let fn_ty = self.extract_function_signature(store, &fn_sig, span);
        self.sink.truncate(before);

        let scope = self.scopes.current_mut();
        scope.values.insert(name.to_string(), fn_ty);
    }
}