litex-lang 0.9.88-beta

The Formal Way to Write Math as It Looks
Documentation
use crate::prelude::*;
use std::collections::HashMap;

impl Runtime {
    // TODO: THIS IS A MESS
    pub fn exec_have_obj_equal_stmt(
        &mut self,
        have_obj_equal_stmt: &HaveObjEqualStmt,
    ) -> Result<StmtResult, RuntimeError> {
        if have_obj_equal_stmt.param_def.number_of_params()
            != have_obj_equal_stmt.objs_equal_to.len()
        {
            return Err(short_exec_error(
 have_obj_equal_stmt.clone().into(),
                    "have_obj_equal_stmt: number of params in param_def does not match number of objs_equal_to".to_string(),
                    None,
                    vec![],
                ));
        }

        let mut current_index = 0;
        let mut param_to_obj_map: HashMap<String, Obj> = HashMap::new();
        for param_def in have_obj_equal_stmt.param_def.groups.iter() {
            let current_type_holder = self
                .inst_param_type(
                    &param_def.param_type,
                    &param_to_obj_map,
                    ParamObjType::Identifier,
                )
                .map_err(|runtime_error| {
                    short_exec_error(
                        have_obj_equal_stmt.clone().into(),
                        "",
                        Some(runtime_error),
                        vec![],
                    )
                })?;
            let current_type = &current_type_holder;
            for name in param_def.params.iter() {
                let current_param_equal_to = &have_obj_equal_stmt.objs_equal_to[current_index];

                let verify_result = self
                    .verify_obj_satisfies_param_type(
                        current_param_equal_to.clone(),
                        current_type,
                        &VerifyState::new(0, false),
                    )
                    .map_err(|verify_error| {
                        short_exec_error(
                            have_obj_equal_stmt.clone().into(),
                            "",
                            Some(verify_error),
                            vec![],
                        )
                    })?;
                if verify_result.is_unknown() {
                    let msg = format!(
                        "have_obj_equal_stmt: {} is not in type {}",
                        current_param_equal_to, current_type
                    );
                    return Err(short_exec_error(
                        have_obj_equal_stmt.clone().into(),
                        msg,
                        None,
                        vec![],
                    ));
                }

                param_to_obj_map.insert(name.clone(), current_param_equal_to.clone());
                current_index += 1;
            }
        }

        let mut infer_result = InferResult::new();

        let param_infer_result = self
            .define_params_with_type(
                &have_obj_equal_stmt.param_def,
                true,
                ParamObjType::Identifier,
            )
            .map_err(|define_params_error| {
                short_exec_error(
                    have_obj_equal_stmt.clone().into(),
                    "",
                    Some(define_params_error),
                    vec![],
                )
            })?;
        infer_result.new_infer_result_inside(param_infer_result);

        for (name, obj) in have_obj_equal_stmt
            .param_def
            .collect_param_names()
            .iter()
            .zip(have_obj_equal_stmt.objs_equal_to.iter())
        {
            let equal_to_fact = EqualFact::new(
                Identifier::new(name.clone()).into(),
                obj.clone(),
                have_obj_equal_stmt.line_file.clone(),
            )
            .into();
            let equal_to_fact_infer_result = self
                .store_atomic_fact_without_well_defined_verified_and_infer(equal_to_fact)
                .map_err(|store_fact_error| {
                    short_exec_error(
                        have_obj_equal_stmt.clone().into(),
                        "",
                        Some(store_fact_error),
                        vec![],
                    )
                })?;
            infer_result.new_infer_result_inside(equal_to_fact_infer_result);
        }

        let lf = have_obj_equal_stmt.line_file.clone();
        for ((name, param_type), obj) in have_obj_equal_stmt
            .param_def
            .collect_param_names_with_types()
            .into_iter()
            .zip(have_obj_equal_stmt.objs_equal_to.iter())
        {
            match (param_type, obj) {
                (ParamType::Obj(Obj::FiniteSeqSet(fs)), Obj::FiniteSeqListObj(list)) => {
                    self.store_known_finite_seq_list_obj(
                        &name,
                        list.clone(),
                        Some(fs.clone()),
                        lf.clone(),
                    );
                }
                (ParamType::Obj(Obj::MatrixSet(ms)), Obj::MatrixListObj(m)) => {
                    self.store_known_matrix_list_obj(
                        &name,
                        m.clone(),
                        Some(ms.clone()),
                        lf.clone(),
                    );
                }
                _ => {}
            }
        }

        Ok(
            (NonFactualStmtSuccess::new(have_obj_equal_stmt.clone().into(), infer_result, vec![]))
                .into(),
        )
    }
}