luaur-analysis 0.1.3

Luau type checker and type inference (Rust).
Documentation
use crate::records::replacer::Replacer;
use crate::records::substitution::Substitution;
use crate::records::tarjan::SubstitutionVtable;
use crate::records::txn_log::TxnLog;
use crate::records::type_arena::TypeArena;
use crate::type_aliases::type_id::TypeId;
use crate::type_aliases::type_pack_id::TypePackId;
use core::ffi::c_void;
use luaur_common::macros::luau_assert::LUAU_ASSERT;
use luaur_common::records::dense_hash_map::DenseHashMap;

fn replacer_is_dirty_ty(owner: *mut c_void, ty: TypeId) -> bool {
    unsafe { (*(owner as *mut Replacer)).is_dirty_type_id(ty) }
}

fn replacer_is_dirty_tp(owner: *mut c_void, tp: TypePackId) -> bool {
    unsafe { (*(owner as *mut Replacer)).is_dirty_type_pack_id(tp) }
}

fn replacer_clean_ty(owner: *mut c_void, ty: TypeId) -> TypeId {
    unsafe { (*(owner as *mut Replacer)).clean_type_id(ty) }
}

fn replacer_clean_tp(owner: *mut c_void, tp: TypePackId) -> TypePackId {
    unsafe { (*(owner as *mut Replacer)).clean_type_pack_id(tp) }
}

fn replacer_found_dirty_ty(owner: *mut c_void, ty: TypeId) {
    unsafe { (*(owner as *mut Replacer)).base.found_dirty_type_id(ty) }
}

fn replacer_found_dirty_tp(owner: *mut c_void, tp: TypePackId) {
    unsafe {
        (*(owner as *mut Replacer))
            .base
            .found_dirty_type_pack_id(tp)
    }
}

fn replacer_ignore_children_ty(owner: *mut c_void, ty: TypeId) -> bool {
    unsafe { (*(owner as *mut Replacer)).ignore_children(ty) }
}

fn replacer_ignore_children_tp(_owner: *mut c_void, _tp: TypePackId) -> bool {
    false
}

impl Replacer {
    pub fn replacer(
        arena: *mut TypeArena,
        replacements: *mut DenseHashMap<TypeId, TypeId>,
        replacement_packs: *mut DenseHashMap<TypePackId, TypePackId>,
    ) -> Self {
        let this = Replacer {
            base: Substitution::substitution_new(TxnLog::empty(), arena),
            replacements,
            replacement_packs,
        };
        LUAU_ASSERT!(this.check_replacement_keys());
        this
    }

    fn install_substitution_vtable(&mut self) {
        let owner = self as *mut Replacer as *mut c_void;
        self.base.base.vtable = SubstitutionVtable {
            owner,
            is_dirty_ty: Some(replacer_is_dirty_ty),
            is_dirty_tp: Some(replacer_is_dirty_tp),
            clean_ty: Some(replacer_clean_ty),
            clean_tp: Some(replacer_clean_tp),
            found_dirty_ty: Some(replacer_found_dirty_ty),
            found_dirty_tp: Some(replacer_found_dirty_tp),
            ignore_children_ty: Some(replacer_ignore_children_ty),
            ignore_children_tp: Some(replacer_ignore_children_tp),
            ignore_children_visit_ty: Some(replacer_ignore_children_ty),
            ignore_children_visit_tp: Some(replacer_ignore_children_tp),
        };
    }

    pub fn substitute_type_id(&mut self, ty: TypeId) -> Option<TypeId> {
        self.install_substitution_vtable();
        self.base.substitute_type_id(ty)
    }

    pub fn substitute_type_pack_id(&mut self, tp: TypePackId) -> Option<TypePackId> {
        self.install_substitution_vtable();
        self.base.substitute_type_pack_id(tp)
    }
}