luaur-analysis 0.1.3

Luau type checker and type inference (Rust).
Documentation
use crate::functions::baseof::baseof;
use crate::records::field::Field;
use crate::records::l_value_hasher::LValueHasher;
use crate::records::symbol::Symbol;
use crate::type_aliases::l_value::{LValue, LValueMember};
use core::hash::{Hash, Hasher};

impl LValueHasher {
    pub fn operator_call(&self, lvalue: &LValue) -> usize {
        // Most likely doesn't produce high quality hashes, but we're probably ok enough with it.
        // When an evidence is shown that operator==(LValue) is used more often than it should, we can have a look at improving the hash quality.
        let mut acc: usize = 0;
        let mut offset: usize = 0;

        let mut current: *const LValue = lvalue;
        unsafe {
            while !current.is_null() {
                if let Some(field) = <Field as LValueMember>::get_if(&*current) {
                    let mut hasher = std::collections::hash_map::DefaultHasher::new();
                    field.key.hash(&mut hasher);
                    let key_hash = hasher.finish() as usize;
                    offset += 1;
                    acc ^= (key_hash << 1) >> offset;
                } else if let Some(symbol) = <Symbol as LValueMember>::get_if(&*current) {
                    acc ^= symbol.hash_luau_symbol_operator_call() << 1;
                } else {
                    debug_assert!(
                        false,
                        "Hash not accumulated for this new LValue alternative."
                    );
                }

                current = baseof(&*current);
            }
        }

        acc
    }
}