* `atomic/1`
* `between/3`
* `call/1..63`
+* `call_with_inference_limit/3`
* `catch/3`
* `compound/1`
* `display/1`
pub enum QueryTerm {
Arg(Vec<Box<Term>>),
CallN(Vec<Box<Term>>),
+ CallWithInferenceLimit(Vec<Box<Term>>),
Catch(Vec<Box<Term>>),
CompareTerm(CompareTermQT, Vec<Box<Term>>),
Cut,
&QueryTerm::Jump(ref vars) => vars.len(),
&QueryTerm::NotEq(_) => 2,
&QueryTerm::CallN(ref terms) => terms.len(),
+ &QueryTerm::CallWithInferenceLimit(_) => 3,
&QueryTerm::Cut => 0,
&QueryTerm::SetupCallCleanup(_) => 3,
&QueryTerm::Term(ref term) => term.arity(),
pub enum ClauseType<'a> {
Arg,
CallN,
+ CallWithInferenceLimit,
Catch,
CompareNumber(CompareNumberQT),
CompareTerm(CompareTermQT),
match self {
&ClauseType::Arg => "arg",
&ClauseType::CallN => "call",
+ &ClauseType::CallWithInferenceLimit => "call_with_inference_limit",
&ClauseType::Catch => "catch",
&ClauseType::CompareNumber(qt) => qt.name(),
&ClauseType::CompareTerm(qt) => qt.name(),
pub enum BuiltInInstruction {
CleanUpBlock,
CompareNumber(CompareNumberQT, ArithmeticTerm, ArithmeticTerm),
+ DefaultTrustMe,
DynamicCompareNumber(CompareNumberQT),
EraseBall,
Fail,
GetBall,
GetCurrentBlock,
GetCutPoint(RegType),
+ InferenceLevel,
InstallCleaner,
+ InstallInferenceCounter(RegType),
InstallNewBlock,
InternalCallN,
IsAtomic(RegType),
IsRational(RegType),
IsString(RegType),
IsVar(RegType),
+ RemoveInferenceCounter(RegType),
ResetBlock,
RestoreCutPolicy,
SetBall,
IsExecute(RegType, ArithmeticTerm),
NotEqCall,
NotEqExecute,
- Proceed,
+ Proceed,
ThrowCall,
ThrowExecute,
}
compare_term_execute!(term_cmp_lt!()), // (@<)/2, 390.
compare_term_execute!(term_cmp_eq!()), // (=@=)/2, 391.
compare_term_execute!(term_cmp_ne!()), // (\=@=)/2, 392.
+ allocate!(4), // call_with_inference_limit/3, 393.
+ fact![get_var_in_fact!(perm_v!(3), 1),
+ get_var_in_fact!(perm_v!(2), 2),
+ get_var_in_fact!(perm_v!(1), 3)],
+ query![put_var!(perm_v!(4), 1)],
+ get_current_block!(),
+ query![put_value!(perm_v!(3), 1),
+ put_value!(perm_v!(2), 2),
+ put_value!(perm_v!(1), 3),
+ put_value!(perm_v!(4), 4)],
+ deallocate!(),
+ goto_execute!(400, 4), // goto call_with_inference_limit/4, 400.
+ try_me_else!(16), // call_with_inference_limit/4, 400.
+ allocate!(7),
+ fact![get_var_in_fact!(perm_v!(6), 1),
+ get_var_in_fact!(perm_v!(7), 2),
+ get_var_in_fact!(perm_v!(4), 3),
+ get_var_in_fact!(perm_v!(3), 4)],
+ query![put_var!(perm_v!(1), 1)],
+ install_new_block!(),
+ install_inference_counter!(perm_v!(7)),
+ get_cp!(perm_v!(5)),
+ query![put_value!(perm_v!(6), 1)],
+ call_n!(1),
+ query![put_var!(perm_v!(2), 3)],
+ remove_inference_counter!(perm_v!(2)),
+ query![put_value!(perm_v!(4), 1),
+ put_value!(perm_v!(5), 2)],
+ inference_level!(),
+ query![put_value!(perm_v!(3), 1),
+ put_value!(perm_v!(1), 2),
+ put_value!(perm_v!(2), 3)],
+ deallocate!(),
+ goto_execute!(436, 3), // goto end_block/3, 436.
+ default_trust_me!(), // 416.
+ allocate!(2),
+ fact![get_var_in_fact!(perm_v!(1), 3)],
+ query![put_value!(temp_v!(4), 1)],
+ reset_block!(),
+ query![put_var!(temp_v!(2), 1)],
+ remove_inference_counter!(temp_v!(1)),
+ query![put_var!(perm_v!(2), 1)],
+ get_ball!(),
+ erase_ball!(),
+ query![put_unsafe_value!(2, 1),
+ put_value!(perm_v!(1), 2)],
+ deallocate!(),
+ goto_execute!(429, 2), // goto handle_ile/2, 429.
+ try_me_else!(5), // handle_ile/2, 429.
+ switch_on_term!(1, 1, 0, 0),
+ fact![get_constant!(atom!("inference_limit_exceeded", atom_tbl), temp_v!(1)),
+ get_constant!(atom!("inference_limit_exceeded", atom_tbl), temp_v!(2))],
+ neck_cut!(),
+ proceed!(),
+ default_trust_me!(),
+ goto_execute!(59, 1), // goto throw/1, 59.
+ try_me_else!(9), // end_block/3, 436.
+ allocate!(1),
+ fact![get_var_in_fact!(perm_v!(1), 1)],
+ query![put_value!(temp_v!(2), 1)],
+ clean_up_block!(),
+ query![put_value!(perm_v!(1), 1)],
+ deallocate!(),
+ reset_block!(),
+ proceed!(),
+ default_trust_me!(),
+ install_inference_counter!(temp_v!(3)),
+ query![put_value!(temp_v!(2), 1)],
+ reset_block!(),
+ fail!()
]
}
code_dir.insert((tabled_rc!("=..", atom_tbl), 2), (PredicateKeyType::BuiltIn, 208));
code_dir.insert((tabled_rc!("length", atom_tbl), 2), (PredicateKeyType::BuiltIn, 261));
- code_dir.insert((tabled_rc!("setup_call_cleanup", atom_tbl), 3), (PredicateKeyType::BuiltIn, 294));
+ code_dir.insert((tabled_rc!("setup_call_cleanup", atom_tbl), 3),
+ (PredicateKeyType::BuiltIn, 294));
+ code_dir.insert((tabled_rc!("call_with_inference_limit", atom_tbl), 3),
+ (PredicateKeyType::BuiltIn, 393));
+ code_dir.insert((tabled_rc!("_handle_inference_limit_exceeded", atom_tbl), 2),
+ (PredicateKeyType::BuiltIn, 421));
+
code_dir.insert((tabled_rc!("compound", atom_tbl), 1), (PredicateKeyType::BuiltIn, 372));
code_dir.insert((tabled_rc!("rational", atom_tbl), 1), (PredicateKeyType::BuiltIn, 374));
code_dir.insert((tabled_rc!("string", atom_tbl), 1), (PredicateKeyType::BuiltIn, 376));
fn add_conditional_call(code: &mut Code, qt: &QueryTerm, pvs: usize)
{
match qt {
+ &QueryTerm::CallWithInferenceLimit(_) =>
+ code.push(goto_call!(393, 3)),
&QueryTerm::SetupCallCleanup(_) =>
code.push(goto_call!(294, 3)),
&QueryTerm::Arg(_) => {
&ControlInstruction::IsCall(r, ref at) =>
write!(f, "is_call {}, {}", r, at),
&ControlInstruction::IsExecute(r, ref at) =>
- write!(f, "is_execute {}, {}", r, at),
+ write!(f, "is_execute {}, {}", r, at),
&ControlInstruction::DynamicIs =>
- write!(f, "call_is"),
+ write!(f, "call_is"),
&ControlInstruction::JmpByCall(arity, offset) =>
write!(f, "jmp_by_call {}/{}", offset, arity),
&ControlInstruction::JmpByExecute(arity, offset) =>
&ControlInstruction::ThrowCall =>
write!(f, "call_throw"),
&ControlInstruction::ThrowExecute =>
- write!(f, "execute_throw"),
+ write!(f, "execute_throw"),
}
}
}
match self {
&BuiltInInstruction::CleanUpBlock =>
write!(f, "clean_up_block"),
+ &BuiltInInstruction::DefaultTrustMe =>
+ write!(f, "default_trust_me"),
+ &BuiltInInstruction::InstallInferenceCounter(r) =>
+ write!(f, "install_inference_counter {}", r),
&BuiltInInstruction::EraseBall =>
write!(f, "erase_ball"),
&BuiltInInstruction::Fail =>
write!(f, "get_current_block X1"),
&BuiltInInstruction::GetCutPoint(r) =>
write!(f, "get_cp {}", r),
+ &BuiltInInstruction::InferenceLevel =>
+ write!(f, "inference_level"),
&BuiltInInstruction::InstallCleaner =>
write!(f, "install_cleaner"),
&BuiltInInstruction::InstallNewBlock =>
write!(f, "number_test {}, {}, {} ", cmp, at_1, at_2),
&BuiltInInstruction::DynamicCompareNumber(cmp) =>
write!(f, "dynamic_number_test {}", cmp),
+ &BuiltInInstruction::RemoveInferenceCounter(r) =>
+ write!(f, "remove_inference_counter {}", r)
}
}
}
let state = TermIterState::Clause(1, ClauseType::CallN, terms);
QueryIterator { state_stack: vec![state] }
},
+ &QueryTerm::CallWithInferenceLimit(ref terms) => {
+ let ct = ClauseType::CallWithInferenceLimit;
+ let state = TermIterState::Clause(0, ct, terms);
+
+ QueryIterator { state_stack: vec![state] }
+ },
&QueryTerm::Catch(ref terms) => {
let state = TermIterState::Clause(0, ClauseType::Catch, terms);
QueryIterator { state_stack: vec![state] }
arity = child_terms.len() + 1;
break;
},
- &QueryTerm::Catch(ref child_terms) | &QueryTerm::Throw(ref child_terms) => {
+ &QueryTerm::Catch(ref child_terms)
+ | &QueryTerm::Throw(ref child_terms) => {
result.push(term);
arity = child_terms.len();
break;
break;
},
&QueryTerm::Arg(_)
- | &QueryTerm::Functor(_) => {
+ | &QueryTerm::Functor(_)
+ | &QueryTerm::CallWithInferenceLimit(_) => {
result.push(term);
arity = 3;
break;
memberchk(X, Xs) :- member(X, Xs), !.
reverse(Xs, Ys) :- var(Ys), !, reverse(Xs, [], Ys).
- reverse(Ys, Xs) :- var(Ys), reverse(Xs, [], Ys).
+ reverse(Ys, Xs) :- reverse(Xs, [], Ys).
reverse([], Ys, Ys).
reverse([H|T], Ps, Rs) :-
use prolog::and_stack::*;
+use prolog::builtins::CodeDir;
use prolog::ast::*;
use prolog::copier::*;
+use prolog::num::{BigInt, BigUint, Zero, One};
use prolog::or_stack::*;
use prolog::tabled_rc::*;
use downcast::Any;
+
+use std::cmp::Ordering;
+use std::collections::binary_heap::BinaryHeap;
+use std::mem::swap;
use std::ops::{Index, IndexMut};
+use std::rc::Rc;
pub(super) struct DuplicateTerm<'a> {
state: &'a mut MachineState
pub(super) interms: Vec<Number>, // intermediate numbers.
}
+pub(crate) type CallResult = Result<(), Vec<HeapCellValue>>;
+
+pub(crate) trait CallPolicy: Any {
+ fn try_call(&mut self, machine_st: &mut MachineState, code_dir: &CodeDir,
+ name: TabledRc<Atom>, arity: usize)
+ -> CallResult
+ {
+ let compiled_tl_index = code_dir.get(&(name, arity)).map(|index| index.1);
+
+ match compiled_tl_index {
+ Some(compiled_tl_index) => {
+ machine_st.cp = machine_st.p + 1;
+ machine_st.num_of_args = arity;
+ machine_st.b0 = machine_st.b;
+ machine_st.p = CodePtr::DirEntry(compiled_tl_index);
+ },
+ None => machine_st.fail = true
+ };
+
+ Ok(())
+ }
+
+ fn try_execute(&mut self, machine_st: &mut MachineState, code_dir: &CodeDir,
+ name: TabledRc<Atom>, arity: usize)
+ -> CallResult
+ {
+ let compiled_tl_index = code_dir.get(&(name, arity)).map(|index| index.1);
+
+ match compiled_tl_index {
+ Some(compiled_tl_index) => {
+ machine_st.num_of_args = arity;
+ machine_st.b0 = machine_st.b;
+ machine_st.p = CodePtr::DirEntry(compiled_tl_index);
+ },
+ None => machine_st.fail = true
+ };
+
+ Ok(())
+ }
+
+ fn retry_me_else(&mut self, machine_st: &mut MachineState, offset: usize) -> CallResult
+ {
+ let b = machine_st.b - 1;
+ let n = machine_st.or_stack[b].num_args();
+
+ for i in 1 .. n + 1 {
+ machine_st.registers[i] = machine_st.or_stack[b][i].clone();
+ }
+
+ machine_st.e = machine_st.or_stack[b].e;
+ machine_st.cp = machine_st.or_stack[b].cp;
+
+ machine_st.or_stack[b].bp = machine_st.p + offset;
+
+ let old_tr = machine_st.or_stack[b].tr;
+ let curr_tr = machine_st.tr;
+
+ machine_st.unwind_trail(old_tr, curr_tr);
+ machine_st.tr = machine_st.or_stack[b].tr;
+
+ machine_st.trail.truncate(machine_st.tr);
+ machine_st.heap.truncate(machine_st.or_stack[b].h);
+
+ machine_st.hb = machine_st.heap.h;
+
+ machine_st.p += 1;
+
+ Ok(())
+ }
+
+ fn retry(&mut self, machine_st: &mut MachineState, offset: usize) -> CallResult
+ {
+ let b = machine_st.b - 1;
+ let n = machine_st.or_stack[b].num_args();
+
+ for i in 1 .. n + 1 {
+ machine_st.registers[i] = machine_st.or_stack[b][i].clone();
+ }
+
+ machine_st.e = machine_st.or_stack[b].e;
+ machine_st.cp = machine_st.or_stack[b].cp;
+
+ machine_st.or_stack[b].bp = machine_st.p + 1;
+
+ let old_tr = machine_st.or_stack[b].tr;
+ let curr_tr = machine_st.tr;
+
+ machine_st.unwind_trail(old_tr, curr_tr);
+ machine_st.tr = machine_st.or_stack[b].tr;
+
+ machine_st.trail.truncate(machine_st.tr);
+ machine_st.heap.truncate(machine_st.or_stack[b].h);
+
+ machine_st.hb = machine_st.heap.h;
+
+ machine_st.p += offset;
+
+ Ok(())
+ }
+
+ fn trust(&mut self, machine_st: &mut MachineState, offset: usize) -> CallResult
+ {
+ let b = machine_st.b - 1;
+ let n = machine_st.or_stack[b].num_args();
+
+ for i in 1 .. n + 1 {
+ machine_st.registers[i] = machine_st.or_stack[b][i].clone();
+ }
+
+ machine_st.e = machine_st.or_stack[b].e;
+ machine_st.cp = machine_st.or_stack[b].cp;
+
+ let old_tr = machine_st.or_stack[b].tr;
+ let curr_tr = machine_st.tr;
+
+ machine_st.unwind_trail(old_tr, curr_tr);
+
+ machine_st.tr = machine_st.or_stack[b].tr;
+ machine_st.trail.truncate(machine_st.tr);
+
+ machine_st.heap.truncate(machine_st.or_stack[b].h);
+ machine_st.b = machine_st.or_stack[b].b;
+
+ machine_st.or_stack.truncate(machine_st.b);
+
+ machine_st.hb = machine_st.heap.h;
+ machine_st.p += offset;
+
+ Ok(())
+ }
+
+ fn trust_me(&mut self, machine_st: &mut MachineState) -> CallResult
+ {
+ let b = machine_st.b - 1;
+ let n = machine_st.or_stack[b].num_args();
+
+ for i in 1 .. n + 1 {
+ machine_st.registers[i] = machine_st.or_stack[b][i].clone();
+ }
+
+ machine_st.e = machine_st.or_stack[b].e;
+ machine_st.cp = machine_st.or_stack[b].cp;
+
+ let old_tr = machine_st.or_stack[b].tr;
+ let curr_tr = machine_st.tr;
+
+ machine_st.unwind_trail(old_tr, curr_tr);
+
+ machine_st.tr = machine_st.or_stack[b].tr;
+ machine_st.trail.truncate(machine_st.tr);
+
+ machine_st.heap.truncate(machine_st.or_stack[b].h);
+
+ machine_st.b = machine_st.or_stack[b].b;
+
+ machine_st.or_stack.truncate(machine_st.b);
+
+ machine_st.hb = machine_st.heap.h;
+ machine_st.p += 1;
+
+ Ok(())
+ }
+}
+
+downcast!(CallPolicy);
+
+pub(crate) struct DefaultCallPolicy {}
+
+impl CallPolicy for DefaultCallPolicy {}
+
+#[derive(Clone, Eq, PartialEq)]
+struct InferenceLimit(BigUint);
+
+// ensure the heap behaves as a min-heap.
+impl Ord for InferenceLimit {
+ fn cmp(&self, other: &InferenceLimit) -> Ordering {
+ other.0.cmp(&self.0)
+ }
+}
+
+impl PartialOrd for InferenceLimit {
+ fn partial_cmp(&self, other: &InferenceLimit) -> Option<Ordering> {
+ Some(self.cmp(other))
+ }
+}
+
+pub(crate) struct CallWithInferenceLimitCallPolicy {
+ atom_tbl: TabledData<Atom>,
+ pub(crate) prev_policy: Box<CallPolicy>,
+ count: BigUint,
+ limits: BinaryHeap<InferenceLimit>
+}
+
+impl CallWithInferenceLimitCallPolicy {
+ pub(crate) fn new_in_place(atom_tbl: TabledData<Atom>, policy: &mut Box<CallPolicy>)
+ {
+ let mut prev_policy: Box<CallPolicy> = Box::new(DefaultCallPolicy {});
+ swap(&mut prev_policy, policy);
+
+ let new_policy = CallWithInferenceLimitCallPolicy { atom_tbl, prev_policy,
+ count: BigUint::zero(),
+ limits: BinaryHeap::new() };
+
+ *policy = Box::new(new_policy);
+ }
+
+ fn increment(&mut self) -> CallResult {
+ if let Some(ref limit) = self.limits.peek() {
+ if self.count == limit.0 {
+ return Err(vec![heap_atom!("inference_limit_exceeded", self.atom_tbl)])
+ } else {
+ self.count = BigUint::one() + &self.count;
+ }
+ }
+
+ Ok(())
+ }
+
+ pub(crate) fn add_limit(&mut self, limit: Rc<BigInt>) {
+ match limit.to_biguint() {
+ Some(limit) => self.limits.push(InferenceLimit(limit + &self.count)),
+ _ => panic!("add_limit: expected unsigned integer.")
+ };
+ }
+
+ pub(crate) fn remove_limit(&mut self) -> Option<BigUint> {
+ self.limits.pop().map(|i| i.0 - &self.count)
+ }
+
+ pub(crate) fn is_empty(&self) -> bool {
+ self.limits.is_empty()
+ }
+
+ pub(crate) fn into_inner(&mut self) -> Box<CallPolicy> {
+ let mut new_inner: Box<CallPolicy> = Box::new(DefaultCallPolicy {});
+ swap(&mut self.prev_policy, &mut new_inner);
+ new_inner
+ }
+}
+
+impl CallPolicy for CallWithInferenceLimitCallPolicy {
+ fn try_call(&mut self, machine_st: &mut MachineState, code_dir: &CodeDir,
+ name: TabledRc<Atom>, arity: usize)
+ -> CallResult
+ {
+ self.prev_policy.try_call(machine_st, code_dir, name, arity)?;
+ self.increment()
+ }
+
+ fn try_execute(&mut self, machine_st: &mut MachineState, code_dir: &CodeDir,
+ name: TabledRc<Atom>, arity: usize)
+ -> CallResult
+ {
+ self.prev_policy.try_execute(machine_st, code_dir, name, arity)?;
+ self.increment()
+ }
+
+ fn retry_me_else(&mut self, machine_st: &mut MachineState, offset: usize) -> CallResult
+ {
+ self.prev_policy.retry_me_else(machine_st, offset)?;
+ self.increment()
+ }
+
+ fn retry(&mut self, machine_st: &mut MachineState, offset: usize) -> CallResult
+ {
+ self.prev_policy.retry(machine_st, offset)?;
+ self.increment()
+ }
+
+ fn trust_me(&mut self, machine_st: &mut MachineState) -> CallResult
+ {
+ self.prev_policy.trust_me(machine_st)?;
+ self.increment()
+ }
+
+ fn trust(&mut self, machine_st: &mut MachineState, offset: usize) -> CallResult
+ {
+ self.prev_policy.trust(machine_st, offset)?;
+ self.increment()
+ }
+}
+
pub(crate) trait CutPolicy: Any {
fn cut(&mut self, &mut MachineState, RegType);
}
return;
}
- machine_st.p += 1;
+ machine_st.p += 1;
}
}
pub(crate) struct SetupCallCleanupCutPolicy {
// locations of cleaners, cut points, the previous block
- cont_pts: Vec<(Addr, usize, usize)>
+ cont_pts: Vec<(Addr, usize, usize)>
}
impl SetupCallCleanupCutPolicy {
pub(crate) fn new() -> Self {
SetupCallCleanupCutPolicy { cont_pts: vec![] }
}
-
+
pub(crate) fn out_of_cont_pts(&self) -> bool {
self.cont_pts.is_empty()
}
}
machine_st.p += 1;
-
+
if !self.out_of_cont_pts() {
machine_st.cp = machine_st.p;
machine_st.num_of_args = 0;
}
}
- fn unwind_trail(&mut self, a1: usize, a2: usize) {
+ pub(super) fn unwind_trail(&mut self, a1: usize, a2: usize) {
for i in a1 .. a2 {
match self.trail[i] {
Ref::HeapCell(r) =>
}
}
- fn try_call_predicate(&mut self, code_dir: &CodeDir, name: TabledRc<Atom>, arity: usize)
- {
- let compiled_tl_index = code_dir.get(&(name, arity)).map(|index| index.1);
-
- match compiled_tl_index {
- Some(compiled_tl_index) => {
- self.cp = self.p + 1;
- self.num_of_args = arity;
- self.b0 = self.b;
- self.p = CodePtr::DirEntry(compiled_tl_index);
- },
- None => self.fail = true
- };
- }
-
- fn try_execute_predicate(&mut self, code_dir: &CodeDir, name: TabledRc<Atom>, arity: usize)
- {
- let compiled_tl_index = code_dir.get(&(name, arity)).map(|index| index.1);
-
- match compiled_tl_index {
- Some(compiled_tl_index) => {
- self.num_of_args = arity;
- self.b0 = self.b;
- self.p = CodePtr::DirEntry(compiled_tl_index);
- },
- None => self.fail = true
- };
- }
-
- fn handle_internal_call_n(&mut self, code_dir: &CodeDir)
+ fn handle_internal_call_n(&mut self, call_policy: &mut Box<CallPolicy>, code_dir: &CodeDir)
{
let arity = self.num_of_args + 1;
let pred = self.registers[1].clone();
self.registers[arity - 1] = pred;
if let Some((name, arity)) = self.setup_call_n(arity - 1) {
- self.try_execute_predicate(code_dir, name, arity);
+ try_or_fail!(self, call_policy.try_execute(self, code_dir, name, arity));
}
} else {
self.fail = true;
}
pub(super) fn execute_built_in_instr(&mut self, code_dir: &CodeDir,
- cut_policy: &mut Box<CutPolicy>,
+ call_policy: &mut Box<CallPolicy>,
+ cut_policy: &mut Box<CutPolicy>,
instr: &BuiltInInstruction)
{
match instr {
self.compare_numbers(cmp, n1, n2);
},
+ &BuiltInInstruction::DefaultTrustMe => {
+ let mut call_policy = DefaultCallPolicy {};
+ try_or_fail!(self, call_policy.trust_me(self));
+ },
&BuiltInInstruction::DynamicCompareNumber(cmp) => {
let n1 = try_or_fail!(self, self.arith_eval_by_metacall(temp_v!(1)));
let n2 = try_or_fail!(self, self.arith_eval_by_metacall(temp_v!(2)));
self.compare_numbers(cmp, n1, n2);
},
+ &BuiltInInstruction::EraseBall => {
+ self.ball.0 = 0;
+ self.ball.1.truncate(0);
+ self.p += 1;
+ },
&BuiltInInstruction::GetArgCall =>
try_or_fail!(self, {
let val = self.try_get_arg();
self.write_constant_to_var(addr, c);
self.p += 1;
},
- &BuiltInInstruction::EraseBall => {
- self.ball.0 = 0;
- self.ball.1.truncate(0);
- self.p += 1;
- },
&BuiltInInstruction::GetBall => {
let addr = self.store(self.deref(self[temp_v!(1)].clone()));
let h = self.heap.h;
self.p += 1;
},
+ &BuiltInInstruction::InferenceLevel => { // X1 = R, X2 = B.
+ let a1 = self[temp_v!(1)].clone();
+ let a2 = self.store(self.deref(self[temp_v!(2)].clone()));
+
+ match a2 {
+ Addr::Con(Constant::Usize(bp)) =>
+ if self.b <= bp {
+ let a2 = Addr::Con(atom!("!", self.atom_tbl));
+ self.unify(a1, a2);
+ } else {
+ let a2 = Addr::Con(atom!("true", self.atom_tbl));
+ self.unify(a1, a2);
+ },
+ _ => self.fail = true
+ };
+
+ self.p += 1;
+ },
&BuiltInInstruction::InstallCleaner => {
let addr = self[temp_v!(1)].clone();
let b = self.b;
self.p += 1;
},
+ &BuiltInInstruction::InstallInferenceCounter(r) => {
+ let addr = self.store(self.deref(self[r].clone()));
+
+ if call_policy.downcast_ref::<CallWithInferenceLimitCallPolicy>().is_err() {
+ CallWithInferenceLimitCallPolicy::new_in_place(self.atom_tbl.clone(), call_policy);
+ }
+
+ self.p += 1;
+
+ match addr {
+ Addr::Con(Constant::Number(Number::Integer(n))) =>
+ match call_policy.downcast_mut::<CallWithInferenceLimitCallPolicy>().ok() {
+ Some(call_policy) => call_policy.add_limit(n),
+ None => panic!("install_inference_counter: should have installed \\
+ CallWithInferenceLimitCallPolicy.")
+ },
+ _ => {
+ let atom_tbl = self.atom_tbl.clone();
+ self.throw_exception(functor!(atom_tbl,
+ "type_error",
+ 1,
+ [heap_atom!("integer_expected", atom_tbl)]))
+ }
+ };
+ },
&BuiltInInstruction::IsAtomic(r) => {
let d = self.store(self.deref(self[r].clone()));
_ => self.fail = true
};
},
+ &BuiltInInstruction::RemoveInferenceCounter(r) => {
+ let restore_default =
+ match call_policy.downcast_mut::<CallWithInferenceLimitCallPolicy>().ok() {
+ Some(call_policy) => {
+ if let Some(diff) = call_policy.remove_limit() {
+ let addr = self[r].clone();
+ self.unify(addr, Addr::Con(integer!(diff)));
+ } else {
+ panic!("remove_inference_counters: no limit found.");
+ }
+
+ if call_policy.is_empty() {
+ Some(call_policy.into_inner())
+ } else {
+ None
+ }
+ },
+ None => panic!("remove_inference_counters: requires \\
+ CallWithInferenceLimitCallPolicy.")
+ };
+
+ if let Some(new_policy) = restore_default {
+ *call_policy = new_policy;
+ }
+
+ self.p += 1;
+ },
&BuiltInInstruction::RestoreCutPolicy => {
let restore_default =
if let Ok(cut_policy) = cut_policy.downcast_ref::<SetupCallCleanupCutPolicy>() {
self.fail = true;
},
&BuiltInInstruction::InternalCallN =>
- self.handle_internal_call_n(code_dir),
+ self.handle_internal_call_n(call_policy, code_dir),
&BuiltInInstruction::Fail => {
self.fail = true;
self.p += 1;
false
}
- pub(super) fn execute_ctrl_instr(&mut self, code_dir: &CodeDir, cut_policy: &mut Box<CutPolicy>,
+ pub(super) fn execute_ctrl_instr(&mut self, code_dir: &CodeDir,
+ call_policy: &mut Box<CallPolicy>,
+ cut_policy: &mut Box<CutPolicy>,
instr: &ControlInstruction)
{
match instr {
self.p = CodePtr::DirEntry(150);
},
&ControlInstruction::Call(ref name, arity, _) =>
- self.try_call_predicate(code_dir, name.clone(), arity),
+ try_or_fail!(self, call_policy.try_call(self, code_dir, name.clone(), arity)),
&ControlInstruction::CatchCall => {
self.cp = self.p + 1;
self.num_of_args = 3;
},
&ControlInstruction::CallN(arity) =>
if let Some((name, arity)) = self.setup_call_n(arity) {
- self.try_call_predicate(code_dir, name, arity);
+ try_or_fail!(self, call_policy.try_call(self, code_dir, name, arity))
},
&ControlInstruction::CheckCpExecute => {
let a = self.store(self.deref(self[temp_v!(2)].clone()));
self.p = self.cp;
},
&ControlInstruction::Execute(ref name, arity) =>
- self.try_execute_predicate(code_dir, name.clone(), arity),
+ try_or_fail!(self, call_policy.try_execute(self, code_dir, name.clone(), arity)),
&ControlInstruction::ExecuteN(arity) =>
if let Some((name, arity)) = self.setup_call_n(arity) {
- self.try_execute_predicate(code_dir, name, arity);
+ try_or_fail!(self, call_policy.try_execute(self, code_dir, name, arity))
},
&ControlInstruction::FunctorCall =>
try_or_fail!(self, {
};
}
- pub(super) fn execute_indexed_choice_instr(&mut self, instr: &IndexedChoiceInstruction)
+ pub(super) fn execute_indexed_choice_instr(&mut self, instr: &IndexedChoiceInstruction,
+ call_policy: &mut Box<CallPolicy>)
{
match instr {
&IndexedChoiceInstruction::Try(l) => {
self.hb = self.heap.h;
self.p += l;
},
- &IndexedChoiceInstruction::Retry(l) => {
- let b = self.b - 1;
- let n = self.or_stack[b].num_args();
-
- for i in 1 .. n + 1 {
- self.registers[i] = self.or_stack[b][i].clone();
- }
-
- self.e = self.or_stack[b].e;
- self.cp = self.or_stack[b].cp;
-
- self.or_stack[b].bp = self.p + 1;
-
- let old_tr = self.or_stack[b].tr;
- let curr_tr = self.tr;
-
- self.unwind_trail(old_tr, curr_tr);
- self.tr = self.or_stack[b].tr;
-
- self.trail.truncate(self.tr);
- self.heap.truncate(self.or_stack[b].h);
-
- self.hb = self.heap.h;
-
- self.p += l;
- },
- &IndexedChoiceInstruction::Trust(l) => {
- let b = self.b - 1;
- let n = self.or_stack[b].num_args();
-
- for i in 1 .. n + 1 {
- self.registers[i] = self.or_stack[b][i].clone();
- }
-
- self.e = self.or_stack[b].e;
- self.cp = self.or_stack[b].cp;
-
- let old_tr = self.or_stack[b].tr;
- let curr_tr = self.tr;
-
- self.unwind_trail(old_tr, curr_tr);
-
- self.tr = self.or_stack[b].tr;
- self.trail.truncate(self.tr);
-
- self.heap.truncate(self.or_stack[b].h);
- self.b = self.or_stack[b].b;
-
- self.or_stack.truncate(self.b);
-
- self.hb = self.heap.h;
- self.p += l;
- },
+ &IndexedChoiceInstruction::Retry(l) =>
+ try_or_fail!(self, call_policy.retry(self, l)),
+ &IndexedChoiceInstruction::Trust(l) =>
+ try_or_fail!(self, call_policy.trust(self, l))
};
}
- pub(super) fn execute_choice_instr(&mut self, instr: &ChoiceInstruction)
+ pub(super) fn execute_choice_instr(&mut self, instr: &ChoiceInstruction,
+ call_policy: &mut Box<CallPolicy>)
{
match instr {
&ChoiceInstruction::TryMeElse(offset) => {
self.hb = self.heap.h;
self.p += 1;
},
- &ChoiceInstruction::RetryMeElse(offset) => {
- let b = self.b - 1;
- let n = self.or_stack[b].num_args();
-
- for i in 1 .. n + 1 {
- self.registers[i] = self.or_stack[b][i].clone();
- }
-
- self.e = self.or_stack[b].e;
- self.cp = self.or_stack[b].cp;
-
- self.or_stack[b].bp = self.p + offset;
-
- let old_tr = self.or_stack[b].tr;
- let curr_tr = self.tr;
-
- self.unwind_trail(old_tr, curr_tr);
- self.tr = self.or_stack[b].tr;
-
- self.trail.truncate(self.tr);
- self.heap.truncate(self.or_stack[b].h);
-
- self.hb = self.heap.h;
-
- self.p += 1;
- },
- &ChoiceInstruction::TrustMe => {
- let b = self.b - 1;
- let n = self.or_stack[b].num_args();
-
- for i in 1 .. n + 1 {
- self.registers[i] = self.or_stack[b][i].clone();
- }
-
- self.e = self.or_stack[b].e;
- self.cp = self.or_stack[b].cp;
-
- let old_tr = self.or_stack[b].tr;
- let curr_tr = self.tr;
-
- self.unwind_trail(old_tr, curr_tr);
-
- self.tr = self.or_stack[b].tr;
- self.trail.truncate(self.tr);
-
- self.heap.truncate(self.or_stack[b].h);
-
- self.b = self.or_stack[b].b;
-
- self.or_stack.truncate(self.b);
-
- self.hb = self.heap.h;
- self.p += 1;
- }
+ &ChoiceInstruction::RetryMeElse(offset) =>
+ try_or_fail!(self, call_policy.retry_me_else(self, offset)),
+ &ChoiceInstruction::TrustMe =>
+ try_or_fail!(self, call_policy.trust_me(self))
}
}
pub struct Machine {
ms: MachineState,
+ call_policy: Box<CallPolicy>,
cut_policy: Box<CutPolicy>,
code: Code,
code_dir: CodeDir,
let atom_tbl = Rc::new(RefCell::new(HashSet::new()));
let (code, code_dir, op_dir) = build_code_dir(atom_tbl.clone());
-
Machine {
ms: MachineState::new(atom_tbl),
+ call_policy: Box::new(DefaultCallPolicy {}),
cut_policy: Box::new(DefaultCutPolicy {}),
code,
code_dir,
&Line::Arithmetic(ref arith_instr) =>
self.ms.execute_arith_instr(arith_instr),
&Line::BuiltIn(ref built_in_instr) =>
- self.ms.execute_built_in_instr(&self.code_dir, &mut self.cut_policy, built_in_instr),
+ self.ms.execute_built_in_instr(&self.code_dir, &mut self.call_policy,
+ &mut self.cut_policy, built_in_instr),
&Line::Choice(ref choice_instr) =>
- self.ms.execute_choice_instr(choice_instr),
+ self.ms.execute_choice_instr(choice_instr, &mut self.call_policy),
&Line::Cut(ref cut_instr) =>
self.ms.execute_cut_instr(cut_instr, &mut self.cut_policy),
&Line::Control(ref control_instr) =>
- self.ms.execute_ctrl_instr(&self.code_dir, &mut self.cut_policy, control_instr),
+ self.ms.execute_ctrl_instr(&self.code_dir, &mut self.call_policy,
+ &mut self.cut_policy, control_instr),
&Line::Fact(ref fact) => {
for fact_instr in fact {
if self.failed() {
&Line::Indexing(ref indexing_instr) =>
self.ms.execute_indexing_instr(&indexing_instr),
&Line::IndexedChoice(ref choice_instr) =>
- self.ms.execute_indexed_choice_instr(choice_instr),
+ self.ms.execute_indexed_choice_instr(choice_instr, &mut self.call_policy),
&Line::Query(ref query) => {
for query_instr in query {
if self.failed() {
CompareTermQT::Equal
)
}
+
+macro_rules! install_inference_counter {
+ ($r:expr) => (
+ Line::BuiltIn(BuiltInInstruction::InstallInferenceCounter($r))
+ )
+}
+
+macro_rules! remove_inference_counter {
+ ($r:expr) => (
+ Line::BuiltIn(BuiltInInstruction::RemoveInferenceCounter($r))
+ )
+}
+
+macro_rules! inference_level {
+ () => (
+ Line::BuiltIn(BuiltInInstruction::InferenceLevel)
+ )
+}
+
+macro_rules! default_trust_me {
+ () => (
+ Line::BuiltIn(BuiltInInstruction::DefaultTrustMe)
+ )
+}
-Subproject commit 5cd21c56d516f934d89b40935487cfda8ad903b4
+Subproject commit 74bb1031f54631bae14aee7d8814c33573e7171d