From a68394b6f2a1fe91c168637e13dd07d87b17f718 Mon Sep 17 00:00:00 2001 From: Mark Thom Date: Tue, 7 Jun 2022 21:45:26 -0600 Subject: [PATCH] simplify detecting undefined goal_ and term_expansion --- build/instructions_template.rs | 4 ++++ src/loader.pl | 19 +++++-------------- src/machine/dispatch.rs | 8 ++++++++ src/machine/system_calls.rs | 33 +++++++++++++++++++++++++++++++++ 4 files changed, 50 insertions(+), 14 deletions(-) diff --git a/build/instructions_template.rs b/build/instructions_template.rs index 0cf0a6b5..336cf395 100644 --- a/build/instructions_template.rs +++ b/build/instructions_template.rs @@ -542,6 +542,8 @@ enum SystemClauseType { DeterministicLengthRundown, #[strum_discriminants(strum(props(Arity = "7", Name = "$http_open")))] HttpOpen, + #[strum_discriminants(strum(props(Arity = "3", Name = "$predicate_defined")))] + PredicateDefined, REPL(REPLCodePtr), } @@ -1665,6 +1667,7 @@ fn generate_instruction_preface() -> TokenStream { &Instruction::CallCpuNow(_) | &Instruction::CallDeterministicLengthRundown(_) | &Instruction::CallHttpOpen(_) | + &Instruction::CallPredicateDefined(_) | &Instruction::CallCurrentTime(_) | &Instruction::CallQuotedToken(_) | &Instruction::CallReadTermFromChars(_) | @@ -1869,6 +1872,7 @@ fn generate_instruction_preface() -> TokenStream { &Instruction::ExecuteCpuNow(_) | &Instruction::ExecuteDeterministicLengthRundown(_) | &Instruction::ExecuteHttpOpen(_) | + &Instruction::ExecutePredicateDefined(_) | &Instruction::ExecuteCurrentTime(_) | &Instruction::ExecuteQuotedToken(_) | &Instruction::ExecuteReadTermFromChars(_) | diff --git a/src/loader.pl b/src/loader.pl index ba036061..4738494a 100644 --- a/src/loader.pl +++ b/src/loader.pl @@ -34,23 +34,13 @@ write_error(Error) :- write('.'). '$print_message_and_fail'(Error) :- - ( ( Error = error(existence_error(procedure, Expansion), Expansion) - ; Error = error(evaluation_error((_:_)/_),Expansion) - ) -> - ( ( Expansion = goal_expansion/2 - ; Expansion = term_expansion/2 - ) -> - true - ; write_error(Error), - nl - ) - ; write_error(Error), - nl - ), + write_error(Error), + nl, '$fail'. expand_term(Term, ExpandedTerm) :- - ( catch('$call'(user:term_expansion(Term, ExpandedTerm0)), + ( '$predicate_defined'(user, term_expansion, 2), + catch('$call'(user:term_expansion(Term, ExpandedTerm0)), E, '$call'(loader:'$print_message_and_fail'(E))) -> ( var(ExpandedTerm0) -> @@ -77,6 +67,7 @@ term_expansion_list([Term|Terms], ExpandedTermsHead, ExpandedTermsTail) :- goal_expansion(Goal, Module, ExpandedGoal) :- ( atom(Module), + '$predicate_defined'(Module, goal_expansion, 2), catch('$call'(Module:goal_expansion(Goal, ExpandedGoal0)), E, '$call'(loader:'$print_message_and_fail'(E))) -> diff --git a/src/machine/dispatch.rs b/src/machine/dispatch.rs index 3664ebee..0e0baade 100644 --- a/src/machine/dispatch.rs +++ b/src/machine/dispatch.rs @@ -4851,6 +4851,14 @@ impl Machine { try_or_throw!(self.machine_st, self.add_non_counted_backtracking()); self.machine_st.p = self.machine_st.cp; } + &Instruction::CallPredicateDefined(_) => { + self.machine_st.fail = !self.predicate_defined(); + self.machine_st.p += 1; + } + &Instruction::ExecutePredicateDefined(_) => { + self.machine_st.fail = !self.predicate_defined(); + self.machine_st.p = self.machine_st.cp; + } } } diff --git a/src/machine/system_calls.rs b/src/machine/system_calls.rs index 6e9e7a1a..9a2cd575 100644 --- a/src/machine/system_calls.rs +++ b/src/machine/system_calls.rs @@ -3771,6 +3771,39 @@ impl Machine { self.machine_st.fail = !self.indices.modules.contains_key(&module_name); } + pub(crate) fn predicate_defined(&self) -> bool { + let module_name = cell_as_atom!(self.machine_st.store(self.machine_st.deref( + self.machine_st.registers[1] + ))); + + let name = cell_as_atom!(self.machine_st.store(self.machine_st.deref( + self.machine_st.registers[2] + ))); + + let arity = match Number::try_from(self.machine_st.store(self.machine_st.deref( + self.machine_st.registers[3] + ))) { + Ok(Number::Fixnum(n)) => n.get_num() as usize, + Ok(Number::Integer(n)) => { + if let Some(n) = n.to_usize() { + n + } else { + return false; + } + } + _ => { + unreachable!() + } + }; + + self.indices.get_predicate_code_index( + name, + arity, + module_name, + ).map(|index| index.local().is_some()) + .unwrap_or(false) + } + #[inline(always)] pub(crate) fn no_such_predicate(&mut self) -> CallResult { let module_name = cell_as_atom!(self.machine_st.store(self.machine_st.deref( -- 2.54.0