rational trees
This commit is contained in:
parent
fa96ffa932
commit
38610c0b0d
212
C/terms.c
212
C/terms.c
@ -144,7 +144,7 @@ typedef struct non_single_struct_t {
|
|||||||
*to_visit_max = to_visit + 1024; \
|
*to_visit_max = to_visit + 1024; \
|
||||||
\
|
\
|
||||||
restart: \
|
restart: \
|
||||||
if (pt0 < pt0_end) { \
|
while (pt0 < pt0_end) { \
|
||||||
register CELL d0; \
|
register CELL d0; \
|
||||||
register CELL *ptd0; \
|
register CELL *ptd0; \
|
||||||
++pt0; \
|
++pt0; \
|
||||||
@ -1225,7 +1225,7 @@ static int max_numbered_var(CELL *pt0, CELL *pt0_end, Int *maxp USES_REGS) {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static Int MaxNumberedVar(Term inp, UInt arity_REGS) {
|
static Int MaxNumberedVar(Term inp, UInt arity PASS_REGS) {
|
||||||
Term t = Deref(inp);
|
Term t = Deref(inp);
|
||||||
|
|
||||||
if (IsPrimitiveTerm(t)) {
|
if (IsPrimitiveTerm(t)) {
|
||||||
@ -1240,74 +1240,53 @@ static Int MaxNumberedVar(Term inp, UInt arity_REGS) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#define BREAK_LOOP(BOTTOM, TOP) (AtomTag | (CELL)to_visit)
|
/**
|
||||||
|
* @pred largest_numbervar( +_Term_, -Max)
|
||||||
|
*
|
||||||
|
* Unify _Max_ with the largest integer _I_ such that `$VAR(I)` is a
|
||||||
|
* sub-term of _Term_.
|
||||||
|
*
|
||||||
|
* This built-in predicate is useful if part of a term has been grounded, and
|
||||||
|
* now you want to ground the full term.
|
||||||
|
*/
|
||||||
|
static Int largest_numbervar(USES_REGS1)
|
||||||
|
{
|
||||||
|
return Yap_unify(MaxNumberedVar(Deref(ARG1), 2 PASS_REGS), ARG2);
|
||||||
|
}
|
||||||
|
|
||||||
#define WALK_CYCLES_IN_TERM(LIST0, STRUCT0) \
|
static Term BREAK_LOOP(int ddep ) {
|
||||||
if (IsPairTerm(d0)) { \
|
Term t0 = MkIntegerTerm (ddep);
|
||||||
if (to_visit + 32 >= to_visit_max) { \
|
return Yap_MkApplTerm(Yap_MkFunctor(Yap_LookupAtom("@^"), 1), 1, &t0);
|
||||||
goto aux_overflow; \
|
}
|
||||||
} \
|
|
||||||
CELL *headp = RepPair(d0); \
|
static Term UNFOLD_LOOP( Term t, Term *b, Term *l) {
|
||||||
if (IsAtomTerm(*headp) && \
|
Term ti = Yap_MkNewApplTerm(FunctorEq, 2);
|
||||||
(CELL *)AtomOfTerm(*headp) >= (CELL *)to_visit0 && \
|
RepAppl(ti)[2] = t;
|
||||||
(CELL *)AtomOfTerm(*headp) < (CELL *)to_visit_max) { \
|
Term o = RepAppl(ti)[1];
|
||||||
LIST0; \
|
HR[0] = ti;
|
||||||
*headp = BREAK_LOOP(ptd0, headp); \
|
HR[1] = *l;
|
||||||
goto restart; \
|
l[0] = AbsPair(HR);
|
||||||
} \
|
if (b!=NULL && *b==TermNil)
|
||||||
to_visit->pt0 = pt0; \
|
b = l;
|
||||||
to_visit->pt0_end = pt0_end; \
|
l = HR+1;
|
||||||
to_visit->ptd0 = headp; \
|
|
||||||
to_visit->d0 = *headp; \
|
HR+=2;
|
||||||
to_visit++; \
|
return o;
|
||||||
d0 = *headp; \
|
}
|
||||||
pt0 = headp; \
|
|
||||||
*pt0 = TermFreeTerm; \
|
|
||||||
pt0_end = headp + 1; \
|
|
||||||
if (pt0 <= pt0_end) \
|
|
||||||
goto list_loop; \
|
|
||||||
} else if (IsApplTerm(d0)) { \
|
|
||||||
register Functor f; \
|
|
||||||
register CELL *ap2; \
|
|
||||||
/* store the terms to visit */ \
|
|
||||||
ap2 = RepAppl(d0); \
|
|
||||||
f = (Functor)(*ap2); \
|
|
||||||
\
|
|
||||||
if (IsExtensionFunctor(f) || IsAtomTerm((CELL)f)) { \
|
|
||||||
\
|
|
||||||
*ap2 = BREAK_LOOP(ptd0, ap2); \
|
|
||||||
goto restart; \
|
|
||||||
} \
|
|
||||||
STRUCT0; \
|
|
||||||
if (to_visit + 32 >= to_visit_max) { \
|
|
||||||
goto aux_overflow; \
|
|
||||||
} \
|
|
||||||
to_visit->pt0 = pt0; \
|
|
||||||
to_visit->pt0_end = pt0_end; \
|
|
||||||
to_visit->ptd0 = ap2; \
|
|
||||||
to_visit->d0 = *ap2; \
|
|
||||||
to_visit++; \
|
|
||||||
\
|
|
||||||
*ap2 = TermFoundVar; \
|
|
||||||
d0 = ArityOfFunctor(f); \
|
|
||||||
pt0 = ap2; \
|
|
||||||
pt0_end = ap2 + d0; \
|
|
||||||
goto restart;\
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static int loops_in_complex_term(CELL *pt0, CELL *pt0_end USES_REGS) {
|
static int loops_in_complex_term(CELL *pt0, CELL *pt0_end, Term *listp, Term *endp USES_REGS) {
|
||||||
int lvl = push_text_stack();
|
int lvl = push_text_stack();
|
||||||
|
|
||||||
struct non_single_struct_t *to_visit = Malloc(
|
struct non_single_struct_t *to_visit = Malloc(
|
||||||
1024 * sizeof(struct non_single_struct_t)),
|
1024 * sizeof(struct non_single_struct_t)),
|
||||||
*to_visit0 = to_visit,
|
*to_visit0 = to_visit,
|
||||||
*to_visit_max = to_visit + 1024;
|
*to_visit_max = to_visit + 1024;
|
||||||
|
|
||||||
to_visit0 = to_visit;
|
to_visit0 = to_visit;
|
||||||
to_visit_max = to_visit0 + 1024;
|
to_visit_max = to_visit0 + 1024;
|
||||||
restart:
|
restart:
|
||||||
if (pt0 < pt0_end) {
|
while (pt0 < pt0_end) {
|
||||||
CELL d0;
|
CELL d0;
|
||||||
CELL *ptd0;
|
CELL *ptd0;
|
||||||
++pt0;
|
++pt0;
|
||||||
@ -1315,49 +1294,127 @@ restart:
|
|||||||
d0 = *ptd0;
|
d0 = *ptd0;
|
||||||
list_loop:
|
list_loop:
|
||||||
deref_head(d0, vars_in_term_unk);
|
deref_head(d0, vars_in_term_unk);
|
||||||
vars_in_term_nvar : {
|
vars_in_term_nvar :
|
||||||
WALK_CYCLES_IN_TERM({}, {});
|
if (IsPairTerm(d0)) {
|
||||||
|
if (to_visit + 32 >= to_visit_max) {
|
||||||
|
goto aux_overflow;
|
||||||
|
}
|
||||||
|
CELL *headp = RepPair(d0);
|
||||||
|
|
||||||
goto restart;
|
d0 = headp[0];
|
||||||
|
if (IsAtomTerm(d0) &&
|
||||||
|
(CELL *)AtomOfTerm(d0) >= (CELL *)to_visit0 &&
|
||||||
|
(CELL *)AtomOfTerm(d0) < (CELL *)to_visit_max) {
|
||||||
|
// LIST0;
|
||||||
|
struct non_single_struct_t *v0 = (struct non_single_struct_t *)AtomOfTerm(d0);
|
||||||
|
if (listp) {
|
||||||
|
*ptd0 = UNFOLD_LOOP(AbsPair(headp), listp, endp);
|
||||||
|
} else {
|
||||||
|
*ptd0 = BREAK_LOOP(to_visit-v0);
|
||||||
|
}
|
||||||
|
|
||||||
|
goto restart;
|
||||||
|
}
|
||||||
|
to_visit->pt0 = pt0;
|
||||||
|
to_visit->pt0_end = pt0_end;
|
||||||
|
to_visit->ptd0 = headp;
|
||||||
|
to_visit->d0 = d0;
|
||||||
|
*headp = MkAtomTerm((AtomEntry*)to_visit);
|
||||||
|
to_visit++;
|
||||||
|
pt0 = headp;
|
||||||
|
pt0_end = pt0 + 1;
|
||||||
|
ptd0 = pt0;
|
||||||
|
goto list_loop;
|
||||||
|
} else if (IsApplTerm(d0)) {
|
||||||
|
register Functor f;
|
||||||
|
register CELL *ap2;
|
||||||
|
/* store the terms to visit */
|
||||||
|
ap2 = RepAppl(d0);
|
||||||
|
f = (Functor)(*ap2);
|
||||||
|
if (IsExtensionFunctor(f)) continue;
|
||||||
|
if (IsAtomTerm((CELL)f)) {
|
||||||
|
|
||||||
|
if (listp) {
|
||||||
|
*ptd0 = UNFOLD_LOOP(AbsAppl(ap2), listp, endp);
|
||||||
|
} else {
|
||||||
|
*ptd0 = BREAK_LOOP(to_visit-(struct non_single_struct_t *)AtomOfTerm(*ap2));
|
||||||
|
}
|
||||||
|
goto restart;
|
||||||
|
}
|
||||||
|
// STRUCT0;
|
||||||
|
if (to_visit + 32 >= to_visit_max) {
|
||||||
|
goto aux_overflow;
|
||||||
|
}
|
||||||
|
to_visit->pt0 = pt0;
|
||||||
|
to_visit->pt0_end = pt0_end;
|
||||||
|
to_visit->ptd0 = ap2;
|
||||||
|
to_visit->d0 = *ap2;
|
||||||
|
*ap2 = MkAtomTerm((AtomEntry*)to_visit);
|
||||||
|
to_visit++;
|
||||||
|
|
||||||
|
pt0 = ap2;
|
||||||
|
pt0_end = ap2 + (ArityOfFunctor(f));
|
||||||
}
|
}
|
||||||
|
goto restart;
|
||||||
|
|
||||||
|
|
||||||
derefa_body(d0, ptd0, vars_in_term_unk, vars_in_term_nvar);
|
derefa_body(d0, ptd0, vars_in_term_unk, vars_in_term_nvar);
|
||||||
|
|
||||||
/* Do we still have compound terms to visit */
|
|
||||||
if (to_visit > to_visit0) {
|
|
||||||
to_visit--;
|
|
||||||
CELL *headp = to_visit->ptd0;
|
|
||||||
pt0 = to_visit->pt0;
|
|
||||||
pt0_end = to_visit->pt0_end;
|
|
||||||
if (IsAtomTerm(*headp) &&
|
|
||||||
(CELL *)AtomOfTerm(*headp) >= (CELL *)to_visit0 &&
|
|
||||||
(CELL *)AtomOfTerm(*headp) < (CELL *)to_visit_max) {
|
|
||||||
*to_visit->ptd0 = to_visit->d0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
goto restart;
|
goto restart;
|
||||||
}
|
}
|
||||||
|
/* Do we still have compound terms to visit */
|
||||||
|
if (to_visit > to_visit0) {
|
||||||
|
to_visit--;
|
||||||
|
|
||||||
|
pt0 = to_visit->pt0;
|
||||||
|
pt0_end = to_visit->pt0_end;
|
||||||
|
CELL *ptd0 = to_visit->ptd0;
|
||||||
|
if (!IsVarTerm(*ptd0))
|
||||||
|
*ptd0 = to_visit->d0;
|
||||||
|
goto restart;
|
||||||
|
}
|
||||||
|
|
||||||
pop_text_stack(lvl);
|
pop_text_stack(lvl);
|
||||||
return 0;
|
return 0;
|
||||||
def_aux_overflow();
|
def_aux_overflow();
|
||||||
}
|
}
|
||||||
|
|
||||||
Term Yap_CheckLoops(Term inp, UInt arity_REGS) {
|
Term Yap_CheckLoops(Term inp, UInt arity, Term *listp, Term *endp USES_REGS) {
|
||||||
Term t = Deref(inp);
|
Term t = Deref(inp);
|
||||||
return t;
|
|
||||||
if (IsPrimitiveTerm(t)) {
|
if (IsVarTerm(t) || IsPrimitiveTerm(t)) {
|
||||||
return t;
|
return t;
|
||||||
} else {
|
} else {
|
||||||
Int res;
|
Int res;
|
||||||
|
|
||||||
res = loops_in_complex_term(&t - 1, &t PASS_REGS) - 1;
|
res = loops_in_complex_term((&t) - 1, &t, listp, endp PASS_REGS);
|
||||||
if (res < 0)
|
if (res < 0)
|
||||||
return -1;
|
return -1;
|
||||||
return t;
|
return t;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/** @pred rational_term_to_tree(? _TI_,- _TF_, ?SubTerms, ?MoreSubterms)
|
||||||
|
|
||||||
|
|
||||||
|
The term _TF_ is a forest representation (without cycles) for
|
||||||
|
the Prolog term _TI_. The term _TF_ is the main term. The
|
||||||
|
difference list _SubTerms_-_MoreSubterms_ stores terms of the
|
||||||
|
form _V=T_, where _V_ is a new variable occuring in _TF_, and
|
||||||
|
_T_ is a copy of a sub-term from _TI_.
|
||||||
|
|
||||||
|
|
||||||
|
*/
|
||||||
|
static Int p_break_rational(USES_REGS1)
|
||||||
|
{
|
||||||
|
Term t = Yap_CopyTerm(Deref(ARG1));
|
||||||
|
Term l = Deref(ARG4), k;
|
||||||
|
return Yap_unify(Yap_CheckLoops(t, 4, &k, &l PASS_REGS), ARG2) && Yap_unify(k, ARG3);
|
||||||
|
}
|
||||||
|
|
||||||
void Yap_InitTermCPreds(void) {
|
void Yap_InitTermCPreds(void) {
|
||||||
|
Yap_InitCPred("rational_term_to_tree", 4, p_break_rational, 0);
|
||||||
Yap_InitCPred("term_variables", 2, p_term_variables, 0);
|
Yap_InitCPred("term_variables", 2, p_term_variables, 0);
|
||||||
Yap_InitCPred("term_variables", 3, p_term_variables3, 0);
|
Yap_InitCPred("term_variables", 3, p_term_variables3, 0);
|
||||||
Yap_InitCPred("$variables_in_term", 3, p_variables_in_term, 0);
|
Yap_InitCPred("$variables_in_term", 3, p_variables_in_term, 0);
|
||||||
@ -1377,4 +1434,5 @@ void Yap_InitTermCPreds(void) {
|
|||||||
Yap_InitCPred("ground", 1, p_ground, SafePredFlag);
|
Yap_InitCPred("ground", 1, p_ground, SafePredFlag);
|
||||||
|
|
||||||
Yap_InitCPred("numbervars", 3, p_numbervars, 0);
|
Yap_InitCPred("numbervars", 3, p_numbervars, 0);
|
||||||
|
Yap_InitCPred("largest_numbervar", 2, largest_numbervar, 0);
|
||||||
}
|
}
|
||||||
|
@ -3847,19 +3847,6 @@ void Yap_InitUtilCPreds(void)
|
|||||||
*/
|
*/
|
||||||
Yap_InitCPred("is_list", 1, p_is_list, SafePredFlag|TestPredFlag);
|
Yap_InitCPred("is_list", 1, p_is_list, SafePredFlag|TestPredFlag);
|
||||||
Yap_InitCPred("$is_list_or_partial_list", 1, p_is_list_or_partial_list, SafePredFlag|TestPredFlag);
|
Yap_InitCPred("$is_list_or_partial_list", 1, p_is_list_or_partial_list, SafePredFlag|TestPredFlag);
|
||||||
Yap_InitCPred("rational_term_to_tree", 4, p_break_rational, 0);
|
|
||||||
/** @pred rational_term_to_tree(? _TI_,- _TF_, ?SubTerms, ?MoreSubterms)
|
|
||||||
|
|
||||||
|
|
||||||
The term _TF_ is a forest representation (without cycles and repeated
|
|
||||||
terms) for the Prolog term _TI_. The term _TF_ is the main term. The
|
|
||||||
difference list _SubTerms_-_MoreSubterms_ stores terms of the form
|
|
||||||
_V=T_, where _V_ is a new variable occuring in _TF_, and _T_ is a copy
|
|
||||||
of a sub-term from _TI_.
|
|
||||||
|
|
||||||
|
|
||||||
*/
|
|
||||||
Yap_InitCPred("term_factorized", 3, p_break_rational3, 0);
|
|
||||||
/** @pred term_factorized(? _TI_,- _TF_, ?SubTerms)
|
/** @pred term_factorized(? _TI_,- _TF_, ?SubTerms)
|
||||||
|
|
||||||
|
|
||||||
|
@ -1084,7 +1084,7 @@ void Yap_plwrite(Term t, StreamDesc *mywrite, int max_depth, int flags,
|
|||||||
struct rewind_term rwt;
|
struct rewind_term rwt;
|
||||||
yhandle_t sls = Yap_CurrentSlot();
|
yhandle_t sls = Yap_CurrentSlot();
|
||||||
int lvl = push_text_stack();
|
int lvl = push_text_stack();
|
||||||
|
|
||||||
if (t == 0)
|
if (t == 0)
|
||||||
return;
|
return;
|
||||||
if (!mywrite) {
|
if (!mywrite) {
|
||||||
@ -1109,9 +1109,9 @@ void Yap_plwrite(Term t, StreamDesc *mywrite, int max_depth, int flags,
|
|||||||
wglb.Write_strings = flags & BackQuote_String_f;
|
wglb.Write_strings = flags & BackQuote_String_f;
|
||||||
// if (!(flags & Ignore_cyclics_f) && false)
|
// if (!(flags & Ignore_cyclics_f) && false)
|
||||||
{
|
{
|
||||||
t = Yap_CheckLoops(t, 1);
|
t = Yap_CheckLoops(t, 1, NULL, NULL PASS_REGS);
|
||||||
}
|
}
|
||||||
/* protect slots for portray */
|
/* protect slots for portray */
|
||||||
writeTerm(t, priority, 1, FALSE, &wglb, &rwt);
|
writeTerm(t, priority, 1, FALSE, &wglb, &rwt);
|
||||||
if (flags & New_Line_f) {
|
if (flags & New_Line_f) {
|
||||||
if (flags & Fullstop_f) {
|
if (flags & Fullstop_f) {
|
||||||
|
@ -445,7 +445,7 @@ bool Yap_isDirectory(const char *FileName);
|
|||||||
extern bool Yap_Exists(const char *f);
|
extern bool Yap_Exists(const char *f);
|
||||||
|
|
||||||
/* terms.c */
|
/* terms.c */
|
||||||
extern Term Yap_CheckLoops(Term inp, UInt arity USES_REGS);
|
extern Term Yap_CheckLoops(Term inp, UInt arity, Term *listp, Term *endp USES_REGS);
|
||||||
extern void Yap_InitTermCPreds(void);
|
extern void Yap_InitTermCPreds(void);
|
||||||
|
|
||||||
/* threads.c */
|
/* threads.c */
|
||||||
|
@ -90,12 +90,10 @@ absolute_file_name__(File,LOpts,TrueFileName) :-
|
|||||||
'$absf_port'(fail, File, TrueFileName, HasSol, OldF, PreviousFileErrors, PreviousVerbose, Expand, Verbose, TakeFirst, FileErrors ).
|
'$absf_port'(fail, File, TrueFileName, HasSol, OldF, PreviousFileErrors, PreviousVerbose, Expand, Verbose, TakeFirst, FileErrors ).
|
||||||
|
|
||||||
|
|
||||||
:- start_low_level_trace.
|
|
||||||
prolog:core_file_name(Name, Opts) -->
|
prolog:core_file_name(Name, Opts) -->
|
||||||
'$file_name'(Name, Opts, E),
|
'$file_name'(Name, Opts, E),
|
||||||
'$suffix'(E, Opts),
|
'$suffix'(E, Opts),
|
||||||
'$glob'(Opts).
|
'$glob'(Opts).
|
||||||
:- stop_low_level_trace.
|
|
||||||
%
|
%
|
||||||
% handle library(lists) or foreign(jpl)
|
% handle library(lists) or foreign(jpl)
|
||||||
%
|
%
|
||||||
|
Reference in New Issue
Block a user