luaur-analysis 0.1.1

Luau type checker and type inference (Rust).
Documentation
use crate::functions::as_mutable_type::as_mutable_type_id;
use crate::functions::follow_type::follow_type_id;
use crate::functions::get_type_alt_j::get_type_id;
use crate::records::blocked_type::BlockedType;
use crate::records::constraint_generator::ConstraintGenerator;
use crate::records::interior_free_types::InteriorFreeTypes;
use crate::type_aliases::scope_ptr_constraint_generator::ScopePtr;
use crate::type_aliases::type_id::TypeId;
use crate::type_aliases::type_variant::TypeVariant;
use alloc::vec::Vec;
use luaur_ast::records::ast_stat_block::AstStatBlock;
use luaur_common::macros::luau_assert::LUAU_ASSERT;

impl ConstraintGenerator {
    // ConstraintGenerator::visitFragmentRoot (ConstraintGenerator.cpp).
    pub fn visit_fragment_root(&mut self, resume_scope: &ScopePtr, block: *mut AstStatBlock) {
        // We prepopulate global data in the resumeScope to avoid writing data into the old modules scopes
        let global_scope = self.global_scope.clone().unwrap();
        self.prepopulate_global_scope_for_fragment_typecheck(&global_scope, resume_scope, block);
        // Pre
        self.interior_free_types.push(InteriorFreeTypes::default());
        self.visit_block_without_child_scope(resume_scope.as_ref() as *const _ as *mut _, block);
        // Post
        self.interior_free_types.pop();

        self.fill_in_inferred_bindings(resume_scope, block);

        if !self.logger.is_null() {
            unsafe {
                (*self.logger).capture_generation_module(self.module.clone().unwrap());
            }
        }

        let local_types_pairs: Vec<(TypeId, Vec<TypeId>)> = self
            .local_types
            .iter()
            .map(|(ty, domain)| (*ty, domain.order.clone()))
            .collect();
        for (ty, domain) in local_types_pairs {
            // FIXME: This isn't the most efficient thing.
            let mut domain_ty = unsafe { (*self.builtin_types).neverType };
            for d in domain {
                let d_followed = unsafe { follow_type_id(d) };
                if d_followed == ty {
                    continue;
                }
                domain_ty = self.simplify_union(
                    resume_scope.clone(),
                    unsafe { (*resume_scope.as_ref()).location },
                    domain_ty,
                    d_followed,
                );
            }

            LUAU_ASSERT!(!unsafe { get_type_id::<BlockedType>(ty) }.is_null());
            unsafe {
                (*as_mutable_type_id(ty)).ty = TypeVariant::Bound(domain_ty);
            }
        }
    }
}