target
}
+ /*
fn collect_var_data<'a>(&mut self, mut iter: ChunkedIterator<'a>) -> ConjunctInfo<'a> {
let mut vs = VariableFixtures::new();
let vs = self.marker.drain_var_data(vs, num_of_chunks);
ConjunctInfo::new(vs, num_of_chunks, has_deep_cut)
}
+ */
fn add_conditional_call(&mut self, code: &mut Code, qt: &QueryTerm, pvs: usize) {
match qt {
Ok(())
}
- fn compile_seq_prelude(&mut self, conjunct_info: &ConjunctInfo, body: &mut Code) {
+ fn compile_seq_prelude(&mut self, var_data: &VarData, body: &mut Code) {
+ /*
if conjunct_info.allocates() {
let perm_vars = conjunct_info.perm_vars();
body.push(Instruction::GetLevel(perm_v!(1)));
}
}
+ */
}
fn compile_cleanup(
}
pub(crate) fn compile_rule(&mut self, rule: &Rule) -> Result<Code, CompilationError> {
- let iter = ChunkedIterator::from_rule(rule);
- let conjunct_info = self.collect_var_data(iter);
+ // let iter = ChunkedIterator::from_rule(rule);
+ // let conjunct_info = self.collect_var_data(iter);
let &Rule {
head: (_, ref args, ref p1),
ref clauses,
+ ref var_data,
} = rule;
let mut code = Code::new();
self.marker.reset_at_head(args);
- self.compile_seq_prelude(&conjunct_info, &mut code);
+ self.compile_seq_prelude(&var_data, &mut code);
let iter = FactIterator::from_rule_head_clause(args);
let mut fact = self.compile_target::<FactInstruction, _>(iter, GenContext::Head);
UnsafeVarMarker::from_fact_vars(safe_vars)
}
- pub(crate) fn compile_fact(&mut self, term: &Term) -> Result<Code, CompilationError> {
+ pub(crate) fn compile_fact(&mut self, fact: &Fact) -> Result<Code, CompilationError> {
self.update_var_count(post_order_iter(term));
- let mut vs = VariableFixtures::new();
+ // let mut vs = VariableFixtures::new();
- vs.mark_vars_in_chunk(post_order_iter(term), term.arity(), GenContext::Head);
+ // vs.mark_vars_in_chunk(post_order_iter(term), term.arity(), GenContext::Head);
- vs.populate_restricting_sets();
- self.marker.drain_var_data(vs, 1);
+ // vs.populate_restricting_sets();
+ // self.marker.drain_var_data(vs, 1);
let mut code = Vec::new();
#[derive(Debug)]
pub(crate) struct DebrayAllocator {
- bindings: IndexMap<Var, VarData, FxBuildHasher>,
+ bindings: IndexMap<Var, VarAlloc, FxBuildHasher>,
arg_c: usize,
temp_lb: usize,
arity: usize, // 0 if not at head.
fn occurs_shallowly_in_head(&self, var: &Var, r: usize) -> bool {
match self.bindings.get(var).unwrap() {
- &VarData::Temp(_, _, ref tvd) => tvd.use_set.contains(&(GenContext::Head, r)),
+ &VarAlloc::Temp(_, _, ref tvd) => tvd.use_set.contains(&(GenContext::Head, r)),
_ => false,
}
}
fn alloc_with_cr(&self, var: &Var) -> usize {
match self.bindings.get(var) {
- Some(&VarData::Temp(_, _, ref tvd)) => {
+ Some(&VarAlloc::Temp(_, _, ref tvd)) => {
for &(_, reg) in tvd.use_set.iter() {
if !self.is_in_use(reg) {
return reg;
fn alloc_with_ca(&self, var: &Var) -> usize {
match self.bindings.get(var) {
- Some(&VarData::Temp(_, _, ref tvd)) => {
+ Some(&VarAlloc::Temp(_, _, ref tvd)) => {
for &(_, reg) in tvd.use_set.iter() {
if !self.is_in_use(reg) {
return reg;
// (GenContext::Last(_), k) is in t_var.use_set.
let tvd = self.bindings.get(t_var).unwrap();
- if let &VarData::Temp(_, _, ref tvd) = tvd {
+ if let &VarAlloc::Temp(_, _, ref tvd) = tvd {
if !tvd.use_set.contains(&(GenContext::Last(chunk_num), k)) {
return Some((t_var.clone(), self.alloc_with_ca(t_var)));
}
match term_loc {
GenContext::Head if !r.is_perm() => r.reg_num() == k,
_ => match self.bindings().get(var).unwrap() {
- &VarData::Temp(_, o, _) if r.reg_num() == k => o == k,
+ &VarAlloc::Temp(_, o, _) if r.reg_num() == k => o == k,
_ => false,
},
}
-use crate::parser::ast::*;
-
use crate::forms::*;
use crate::instructions::*;
-use crate::iterators::*;
+use crate::machine::disjuncts::ClassifyInfo;
+use crate::parser::ast::*;
use bit_set::*;
use indexmap::{IndexMap, IndexSet};
-use std::cell::Cell;
-use std::collections::BTreeSet;
-use std::mem::swap;
-use std::vec::Vec;
-
pub(crate) type OccurrenceSet = IndexSet<(GenContext, usize)>;
#[derive(Debug)]
temp_var_data: TempVarData,
}
-// TODO: get ridda this! I think.
// Perm: 0 initially, a stack register once processed.
// Temp: labeled with chunk_num and temp offset (unassigned if 0).
#[derive(Debug)]
-pub(crate) enum VarData {
+pub(crate) enum VarAlloc {
Perm(usize),
Temp(usize, usize, TempVarData),
}
-impl VarData {
+impl VarAlloc {
pub(crate) fn as_reg_type(&self) -> RegType {
match self {
- &VarData::Temp(_, r, _) => RegType::Temp(r),
- &VarData::Perm(r) => RegType::Perm(r),
+ &VarAlloc::Temp(_, r, _) => RegType::Temp(r),
+ &VarAlloc::Perm(r) => RegType::Perm(r),
}
}
}
pub(crate) fn new(last_term_arity: usize) -> Self {
TempVarData {
last_term_arity: last_term_arity,
- use_set: BitSet::new(),
+ use_set: BitSet::<usize>::new(),
no_use_set: BitSet::new(),
conflict_set: BitSet::new(),
}
let mut conflict_set: BitSet<usize> = (1..arity).collect();
for &(_, reg) in self.use_set.iter() {
- conflict_set.remove(®);
+ conflict_set.remove(reg);
}
self.conflict_set = conflict_set;
#[derive(Debug)]
pub(crate) struct VariableFixtures {
temp_vars: IndexMap<usize, TempVarStatus>,
- last_chunk_temp_vars: IndexSet<usize>, // TODO: has no use at all! remove it.
}
-impl<'a> VariableFixtures<'a> {
+impl VariableFixtures {
pub(crate) fn new() -> Self {
VariableFixtures {
temp_vars: IndexMap::new(),
- last_chunk_temp_vars: IndexSet::new(),
}
}
- // TODO: get rid of this also.
- pub(crate) fn insert(&mut self, var: Var, vs: VariableFixture<'a>) {
- self.temp_vars.insert(var, vs);
- }
-
- // TODO: used?
- pub(crate) fn insert_last_chunk_temp_var(&mut self, var: Var) {
- self.last_chunk_temp_vars.insert(var);
- }
-
// computes no_use and conflict sets for all temp vars.
pub(crate) fn populate_restricting_sets(&mut self) {
// three stages:
let TempVarStatus { ref mut temp_var_data, .. } = var_status;
let mut use_set = OccurrenceSet::new();
- mem::swap(&mut temp_var_data.use_set, &mut use_set);
+ std::mem::swap(&mut temp_var_data.use_set, &mut use_set);
use_sets.insert(var_gen_index, use_set);
}
use crate::arena::*;
use crate::atom_table::*;
use crate::instructions::*;
-use crate::machine::disjuncts::VarRecord;
+use crate::machine::disjuncts::VarData;
use crate::machine::heap::*;
use crate::machine::loader::PredicateQueue;
use crate::machine::machine_errors::*;
}
}
-#[derive(Debug, Clone, Copy, PartialEq, Eq)]
+#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum Level {
Deep,
Root,
Counted,
}
-#[derive(Debug, Clone)]
+#[derive(Debug)]
pub enum QueryTerm {
// register, clause type, subterms, clause call policy.
Clause(Cell<RegType>, ClauseType, Vec<Term>, CallPolicy),
#[derive(Debug, Clone)]
pub struct Fact {
pub(crate) head: Term,
- pub(crate) var_records: Vec<VarRecord>,
+ pub(crate) var_data: VarData,
}
#[derive(Debug, Clone)]
pub struct Rule {
pub(crate) head: (Atom, Vec<Term>, QueryTerm),
pub(crate) clauses: Vec<QueryTerm>,
- pub(crate) var_records: Vec<VarRecord>,
+ pub(crate) var_data: VarData,
}
#[derive(Clone, Debug, Hash)]
#[derive(Debug, Clone)]
pub enum PredicateClause {
- Fact(Term),
+ Fact(Fact),
Rule(Rule),
}
#[derive(Debug, Clone)]
pub(crate) enum TermRef<'a> {
AnonVar(Level),
+ Cut(Level),
+ GetLevel(Level),
Cons(Level, &'a Cell<RegType>, &'a Term, &'a Term),
+ Fail(Level),
Literal(Level, &'a Cell<RegType>, &'a Literal),
Clause(Level, &'a Cell<RegType>, Atom, &'a Vec<Term>),
PartialString(Level, &'a Cell<RegType>, &'a String, &'a Box<Term>),
CompleteString(Level, &'a Cell<RegType>, Atom),
Var(Level, &'a Cell<VarReg>, Var),
+ InitialBranch(Level),
+ MiddleBranch(Level),
+ FinalBranch(Level),
}
impl<'a> TermRef<'a> {
pub(crate) fn level(self) -> Level {
match self {
- TermRef::AnonVar(lvl)
- | TermRef::Cons(lvl, ..)
- | TermRef::Literal(lvl, ..)
- | TermRef::Var(lvl, ..)
- | TermRef::Clause(lvl, ..)
- | TermRef::CompleteString(lvl, ..)
- | TermRef::PartialString(lvl, ..) => lvl,
+ TermRef::AnonVar(lvl) |
+ TermRef::Cons(lvl, ..) |
+ TermRef::Cut(lvl) |
+ TermRef::GetLevel(lvl) |
+ TermRef::Literal(lvl, ..) |
+ TermRef::Var(lvl, ..) |
+ TermRef::Clause(lvl, ..) |
+ TermRef::CompleteString(lvl, ..) |
+ TermRef::PartialString(lvl, ..) |
+ TermRef::InitialBranch(lvl) |
+ TermRef::MiddleBranch(lvl) |
+ TermRef::FinalBranch(lvl) |
+ TermRef::Fail(lvl) => lvl,
}
}
}
#[derive(Debug)]
pub(crate) enum TermIterState<'a> {
AnonVar(Level),
- Literal(Level, &'a Cell<RegType>, &'a Literal),
Clause(Level, usize, &'a Cell<RegType>, Atom, &'a Vec<Term>),
+ Cut(Level),
+ Fail(Level),
+ GetLevel(Level),
+ InitialBranch(Level, &'a Vec<QueryTerm>),
+ MiddleBranch(Level, &'a Vec<QueryTerm>),
+ FinalBranch(Level, &'a Vec<QueryTerm>),
+ Sequence(Level, &'a Vec<QueryTerm>),
+ Literal(Level, &'a Cell<RegType>, &'a Literal),
InitialCons(Level, &'a Cell<RegType>, &'a Term, &'a Term),
FinalCons(Level, &'a Cell<RegType>, &'a Term, &'a Term),
InitialPartialString(Level, &'a Cell<RegType>, &'a String, &'a Box<Term>),
FinalPartialString(Level, &'a Cell<RegType>, &'a String, &'a Box<Term>),
CompleteString(Level, &'a Cell<RegType>, Atom),
- UnblockedCut(Level, &'a Cell<VarReg>),
Var(Level, &'a Cell<VarReg>, VarPtr),
}
impl<'a> QueryIterator<'a> {
fn push_subterm(&mut self, lvl: Level, term: &'a Term) {
- self.state_stack
- .push(TermIterState::subterm_to_state(lvl, term));
+ self.state_stack.push(TermIterState::subterm_to_state(lvl, term));
}
fn from_rule_head_clause(terms: &'a Vec<Term>) -> Self {
}
}
- fn new(term: &'a QueryTerm) -> Self {
+ fn extend_state(&mut self, lvl: Level, term: &'a QueryTerm) {
match term {
&QueryTerm::Clause(ref cell, ClauseType::CallN(_), ref terms, _) => {
- let state = TermIterState::Clause(Level::Root, 1, cell, atom!("$call"), terms);
- QueryIterator {
- state_stack: vec![state],
- }
+ self.state_stack.push(TermIterState::Clause(lvl, 1, cell, atom!("$call"), terms));
}
&QueryTerm::Clause(ref cell, ref ct, ref terms, _) => {
- let state = TermIterState::Clause(Level::Root, 0, cell, ct.name(), terms);
- QueryIterator {
- state_stack: vec![state],
- }
+ self.state_stack.push(TermIterState::Clause(lvl, 0, cell, ct.name(), terms));
}
- &QueryTerm::UnblockedCut(ref cell) => {
- let state = TermIterState::UnblockedCut(Level::Root, cell);
-
- QueryIterator {
- state_stack: vec![state],
- }
+ &QueryTerm::Cut => {
+ self.state_stack.push(TermIterState::Cut(lvl));
}
&QueryTerm::GetLevelAndUnify(ref cell, ref var) => {
- let state = TermIterState::Var(Level::Root, cell, VarPtr::from(var));
- QueryIterator {
- state_stack: vec![state],
- }
+ // TODO: get rid of it if possible. or! specialized TermIterState variant.
+ self.state_stack.push(TermIterState::Var(lvl, cell, VarPtr::from(var)));
+ }
+ &QueryTerm::Not(ref terms) => {
+ self.state_stack.push(TermIterState::Fail(lvl));
+ self.state_stack.push(TermIterState::Cut(lvl));
+ self.state_stack.push(TermIterState::Sequence(lvl, terms));
}
- &QueryTerm::Jump(ref vars) => {
- let state_stack = vars
+ &QueryTerm::IfThen(ref if_terms, ref then_terms) => {
+ self.state_stack.push(TermIterState::Sequence(lvl, then_terms));
+ self.state_stack.push(TermIterState::Cut(lvl));
+ self.state_stack.push(TermIterState::Sequence(lvl, if_terms));
+ self.state_stack.push(TermIterState::GetLevel(lvl));
+ }
+ &QueryTerm::Branch(ref branches) => {
+ let len = branches.len();
+ self.state_stack.push(TermIterState::FinalBranch(lvl, &branches[len - 1]));
+
+ self.state_stack.extend(branches[1 .. len - 1]
.iter()
.rev()
- .map(|t| TermIterState::subterm_to_state(Level::Shallow, t))
- .collect();
+ .map(|t| TermIterState::MiddleBranch(lvl, t)),
+ );
- QueryIterator { state_stack }
+ self.state_stack.push(TermIterState::InitialBranch(lvl, &branches[0]));
}
- &QueryTerm::BlockedCut => QueryIterator {
- state_stack: vec![],
- },
}
}
+
+ fn new(term: &'a QueryTerm) -> Self {
+ let mut iter = QueryIterator { state_stack: vec![] };
+ iter.extend_state(Level::Root, term);
+ iter
+ }
}
impl<'a> Iterator for QueryIterator<'a> {
TermIterState::Var(lvl, cell, var) => {
return Some(TermRef::Var(lvl, cell, Var::from(var)));
}
- TermIterState::UnblockedCut(lvl, cell) => {
- return Some(TermRef::Var(lvl, cell, Var::from("!")));
+ TermIterState::Cut(lvl) => {
+ return Some(TermRef::Cut(lvl));
+ }
+ TermIterState::GetLevel(lvl) => {
+ return Some(TermRef::GetLevel(lvl));
+ }
+ TermIterState::InitialBranch(lvl, ref branch) => {
+ self.state_stack.push(TermIterState::Sequence(lvl, branch));
+ return Some(TermRef::InitialBranch(lvl));
+ }
+ TermIterState::MiddleBranch(lvl, ref branch) => {
+ self.state_stack.push(TermIterState::Sequence(lvl, branch));
+ return Some(TermRef::MiddleBranch(lvl));
+ }
+ TermIterState::FinalBranch(lvl, ref branch) => {
+ self.state_stack.push(TermIterState::Sequence(lvl, branch));
+ return Some(TermRef::FinalBranch(lvl));
+ }
+ TermIterState::Sequence(lvl, ref terms) => {
+ for term in branch.iter().rev() {
+ self.extend_state(lvl, term);
+ }
+ }
+ TermIterState::Fail(lvl) => {
+ return Some(TermRef::Fail(lvl));
}
};
}
}
}
-fn contains_cut_var<'a, Iter: Iterator<Item = &'a Term>>(terms: Iter) -> bool {
- for term in terms {
- if let &Term::Var(_, ref var) = term {
- if var.as_str() == Some("!") {
- return true;
- }
- }
- }
-
- false
-}
-
pub(crate) struct ChunkedIterator<'a> {
pub(crate) chunk_num: usize,
iter: Box<dyn Iterator<Item = ChunkedTerm<'a>> + 'a>,
- deep_cut_encountered: bool,
- cut_var_in_head: bool,
}
impl<'a> fmt::Debug for ChunkedIterator<'a> {
.field("chunk_num", &self.chunk_num)
// Hacky solution.
.field("iter", &"Box<dyn Iterator<Item = ChunkedTerm<'a>> + 'a>")
- .field("deep_cut_encountered", &self.deep_cut_encountered)
- .field("cut_var_in_head", &self.cut_var_in_head)
.finish()
}
}
ChunkedIterator {
chunk_num: 0,
iter: Box::new(iter),
- deep_cut_encountered: false,
- cut_var_in_head: false,
}
}
let &Rule {
head: (ref name, ref args, ref p1),
ref clauses,
+ ..
} = rule;
let iter = once(ChunkedTerm::HeadClause(name.clone(), args));
ChunkedIterator {
chunk_num: 0,
iter: Box::new(iter),
- deep_cut_encountered: false,
- cut_var_in_head: false,
}
}
- pub(crate) fn encountered_deep_cut(&self) -> bool {
- self.deep_cut_encountered
- }
-
fn take_chunk(&mut self, term: ChunkedTerm<'a>) -> (usize, usize, Vec<ChunkedTerm<'a>>) {
let mut arity = 0;
let mut item = Some(term);
while let Some(term) = item {
match term {
ChunkedTerm::HeadClause(_, terms) => {
- if contains_cut_var(terms.iter()) {
- self.cut_var_in_head = true;
- }
-
result.push(term);
}
- ChunkedTerm::BodyTerm(&QueryTerm::Jump(ref vars)) => {
+ ChunkedTerm::BodyTerm(&QueryTerm::Cut) => {
result.push(term);
- arity = vars.len();
-
- if contains_cut_var(vars.iter()) && !self.cut_var_in_head {
- self.deep_cut_encountered = true;
- }
-
- break;
- }
- ChunkedTerm::BodyTerm(&QueryTerm::BlockedCut) => {
- result.push(term);
-
- if self.chunk_num > 0 {
- self.deep_cut_encountered = true;
- }
}
ChunkedTerm::BodyTerm(&QueryTerm::GetLevelAndUnify(..)) => {
- self.deep_cut_encountered = true;
-
result.push(term);
arity = 1;
break;
}
- ChunkedTerm::BodyTerm(&QueryTerm::UnblockedCut(..)) => {
- self.deep_cut_encountered = true;
- result.push(term);
- }
ChunkedTerm::BodyTerm(&QueryTerm::Clause(_, ClauseType::Inlined(_), ..)) => {
- result.push(term)
+ result.push(term);
}
ChunkedTerm::BodyTerm(&QueryTerm::Clause(
_,
}
}
+/*
pub(super) fn compile_appendix(
code: &mut Code,
mut queue: VecDeque<TopLevel>,
Ok(())
}
+*/
fn lower_bound_of_target_clause(skeleton: &PredicateSkeleton, target_pos: usize) -> usize {
if target_pos == 0 {
let mut preprocessor = Preprocessor::new(settings);
let clause = self.try_term_to_tl(term, &mut preprocessor)?;
- let queue = preprocessor.parse_queue(self)?;
+ // let queue = preprocessor.parse_queue(self)?;
let mut cg = CodeGenerator::new(
&mut LS::machine_st(&mut self.payload).atom_tbl,
let mut clause_code = cg.compile_predicate(&vec![clause])?;
+ /*
compile_appendix(
&mut clause_code,
queue,
settings.non_counted_bt,
cg.atom_tbl,
)?;
+ */
Ok(StandaloneCompileResult {
clause_code,
clauses.push(self.try_term_to_tl(term, &mut preprocessor)?);
}
- let queue = preprocessor.parse_queue(self)?;
+ // let queue = preprocessor.parse_queue(self)?;
let mut cg = CodeGenerator::new(
&mut LS::machine_st(&mut self.payload).atom_tbl,
let mut code = cg.compile_predicate(&clauses)?;
+ /*
compile_appendix(
&mut code,
queue,
settings.non_counted_bt,
cg.atom_tbl,
)?;
+ */
if settings.is_extensible {
let mut clause_clause_locs = VecDeque::new();
Ok(())
}
+ // TODO: maybe replace Vec<QueryTerm> with an iterator that has, in the stream,
+ // with a 'QueryTerm' that toggles the chunk num and type, like we do here.
fn classify_body_variables<'a, LS: LoadState<'a>>(
&mut self,
loader: &mut Loader<'a, LS>,
debug_assert_eq!(var_data.records.len(), var_num);
+ var_data.fixtures.populate_restricting_sets();
var_data
}
}
}
pub(crate) type HeapVarDict = IndexMap<Var, HeapCellValue, FxBuildHasher>;
-pub(crate) type AllocVarDict = IndexMap<Var, VarData, FxBuildHasher>;
+pub(crate) type AllocVarDict = IndexMap<Var, VarAlloc, FxBuildHasher>;
pub(crate) type GlobalVarDir = IndexMap<Atom, (Ball, Option<HeapCellValue>), FxBuildHasher>;
self.settings.default_call_policy(),
);
- let (head, var_records) = classifier.classify_fact(term)?;
+ let (head, var_data) = classifier.classify_fact(term)?;
- Ok(Fact { head, var_records })
+ Ok(Fact { head, var_data })
}
_ => Err(CompilationError::InadmissibleFact),
}
self.settings.default_call_policy(),
);
- let (head, mut query_terms, var_records) =
+ let (head, mut query_terms, var_data) =
classifier.classify_rule(loader, head, body)?;
let clauses = query_terms.drain(1..).collect();
Term::Clause(_, name, terms) => Ok(Rule {
head: (name, terms, qt),
clauses,
- var_records,
+ var_data,
}),
Term::Literal(_, Literal::Atom(name)) => Ok(Rule {
head: (name, vec![], qt),
clauses,
- var_records,
+ var_data,
}),
_ => Err(CompilationError::InvalidRuleHead),
}