8402df6bb2
git-svn-id: https://yap.svn.sf.net/svnroot/yap/trunk@1135 b08c6af1-5177-4d33-ba66-4b1c6b8b522a
721 lines
17 KiB
C
721 lines
17 KiB
C
/*************************************************************************
|
|
* *
|
|
* YAP Prolog *
|
|
* *
|
|
* Yap Prolog was developed at NCCUP - Universidade do Porto *
|
|
* *
|
|
* Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997 *
|
|
* *
|
|
**************************************************************************
|
|
* *
|
|
* File: attvar.c *
|
|
* Last rev: *
|
|
* mods: *
|
|
* comments: YAP support for attributed vars *
|
|
* *
|
|
*************************************************************************/
|
|
#ifdef SCCS
|
|
static char SccsId[]="%W% %G%";
|
|
#endif
|
|
|
|
#include "Yap.h"
|
|
|
|
#include "Yatom.h"
|
|
#include "Heap.h"
|
|
#include "heapgc.h"
|
|
#include "attvar.h"
|
|
#ifndef NULL
|
|
#define NULL (void *)0
|
|
#endif
|
|
|
|
#ifdef COROUTINING
|
|
|
|
STATIC_PROTO(Term InitVarTime, (void));
|
|
STATIC_PROTO(Int PutAtt, (attvar_record *,Int,Term));
|
|
STATIC_PROTO(Int BuildNewAttVar, (Term,Int,Term));
|
|
|
|
static CELL *
|
|
AddToQueue(attvar_record *attv)
|
|
{
|
|
Term t[2];
|
|
Term WGs, ng;
|
|
|
|
t[0] = (CELL)&(attv->Done);
|
|
t[1] = attv->Value;
|
|
/* follow the chain */
|
|
WGs = Yap_ReadTimedVar(WokenGoals);
|
|
ng = Yap_MkApplTerm(FunctorAttGoal, 2, t);
|
|
|
|
Yap_UpdateTimedVar(WokenGoals, MkPairTerm(ng, WGs));
|
|
if ((Term)WGs == TermNil) {
|
|
/* from now on, we have to start waking up goals */
|
|
Yap_signal(YAP_WAKEUP_SIGNAL);
|
|
}
|
|
return(RepAppl(ng)+2);
|
|
}
|
|
|
|
static void
|
|
AddFailToQueue(void)
|
|
{
|
|
Term WGs;
|
|
|
|
/* follow the chain */
|
|
WGs = Yap_ReadTimedVar(WokenGoals);
|
|
|
|
Yap_UpdateTimedVar(WokenGoals, MkPairTerm(MkAtomTerm(AtomFail),WGs));
|
|
if ((Term)WGs == TermNil) {
|
|
/* from now on, we have to start waking up goals */
|
|
Yap_signal(YAP_WAKEUP_SIGNAL);
|
|
}
|
|
}
|
|
|
|
static int
|
|
CopyAttVar(CELL *orig, CELL ***to_visit_ptr, CELL *res)
|
|
{
|
|
register attvar_record *attv = (attvar_record *)orig;
|
|
register attvar_record *newv;
|
|
CELL **to_visit = *to_visit_ptr;
|
|
Term time = InitVarTime();
|
|
Int j;
|
|
|
|
/* add a new attributed variable */
|
|
newv = (attvar_record *)Yap_ReadTimedVar(DelayedVars);
|
|
if (H0 - (CELL *)newv < 1024+(2*NUM_OF_ATTS))
|
|
return FALSE;
|
|
RESET_VARIABLE(&(newv->Done));
|
|
newv->sus_id = attvars_ext;
|
|
RESET_VARIABLE(&(newv->Value));
|
|
newv->NS = Yap_UpdateTimedVar(AttsMutableList, (CELL)&(newv->Done));
|
|
for (j = 0; j < NUM_OF_ATTS; j++) {
|
|
Term t = Deref(attv->Atts[2*j+1]);
|
|
newv->Atts[2*j] = time;
|
|
|
|
if (IsVarTerm(t)) {
|
|
CELL *vt = VarOfTerm(t);
|
|
if (vt == attv->Atts+(2*j+1)) {
|
|
RESET_VARIABLE(newv->Atts+(2*j+1));
|
|
} else {
|
|
to_visit[0] = vt-1;
|
|
to_visit[1] = vt;
|
|
to_visit[2] = newv->Atts+2*j+1;
|
|
to_visit[3] = vt;
|
|
to_visit += 4;
|
|
}
|
|
} else if (IsVarTerm(t) && IsAtomicTerm(t)) {
|
|
newv->Atts[2*j+1] = t;
|
|
} else {
|
|
to_visit[0] = attv->Atts+2*j;
|
|
to_visit[1] = attv->Atts+2*j+1;
|
|
to_visit[2] = newv->Atts+2*j+1;
|
|
to_visit[3] = (CELL *)(attv->Atts[2*j]);
|
|
to_visit += 4;
|
|
}
|
|
}
|
|
*to_visit_ptr = to_visit;
|
|
*res = (CELL)&(newv->Done);
|
|
Yap_UpdateTimedVar(DelayedVars, (CELL)(newv->Atts+2*j));
|
|
return(TRUE);
|
|
}
|
|
|
|
static Term
|
|
AttVarToTerm(CELL *orig)
|
|
{
|
|
register attvar_record *attv = (attvar_record *)orig;
|
|
Term list = TermNil;
|
|
int j;
|
|
for (j = 0; j < NUM_OF_ATTS; j++) {
|
|
Term t = attv->Atts[2*(NUM_OF_ATTS-j-1)+1];
|
|
if (IsVarTerm(t))
|
|
list = MkPairTerm(MkVarTerm(),list);
|
|
else
|
|
list = MkPairTerm(t,list);
|
|
}
|
|
return(list);
|
|
}
|
|
|
|
static int
|
|
TermToAttVar(Term attvar, Term to)
|
|
{
|
|
return(BuildNewAttVar(to, -1, attvar));
|
|
}
|
|
|
|
static void
|
|
WakeAttVar(CELL* pt1, CELL reg2)
|
|
{
|
|
|
|
/* if bound to someone else, follow until we find the last one */
|
|
attvar_record *attv = (attvar_record *)pt1;
|
|
CELL *myH = H;
|
|
CELL *bind_ptr;
|
|
|
|
if (IsVarTerm(reg2)) {
|
|
if (pt1 == VarOfTerm(reg2))
|
|
return;
|
|
if (IsAttachedTerm(reg2)) {
|
|
attvar_record *susp2 = (attvar_record *)VarOfTerm(reg2);
|
|
|
|
/* binding two suspended variables, be careful */
|
|
if (susp2->sus_id != attvars_ext) {
|
|
/* joining two different kinds of suspensions */
|
|
Yap_Error(SYSTEM_ERROR, TermNil, "joining two different suspensions not implemented");
|
|
return;
|
|
}
|
|
if (susp2 >= attv) {
|
|
if (!IsVarTerm(susp2->Value) || !IsUnboundVar(susp2->Value)) {
|
|
/* oops, our goal is on the queue to be woken */
|
|
if (!Yap_unify(susp2->Value, (CELL)pt1)) {
|
|
AddFailToQueue();
|
|
}
|
|
}
|
|
Bind_Global(&(susp2->Value), (CELL)pt1);
|
|
AddToQueue(susp2);
|
|
return;
|
|
}
|
|
} else {
|
|
Bind(VarOfTerm(reg2), (CELL)pt1);
|
|
return;
|
|
}
|
|
}
|
|
if (!IsVarTerm(attv->Value) || !IsUnboundVar(attv->Value)) {
|
|
/* oops, our goal is on the queue to be woken */
|
|
if (!Yap_unify(attv->Value, reg2)) {
|
|
AddFailToQueue();
|
|
}
|
|
return;
|
|
}
|
|
bind_ptr = AddToQueue(attv);
|
|
if (IsNonVarTerm(reg2)) {
|
|
if (IsPairTerm(reg2) && RepPair(reg2) == myH)
|
|
reg2 = AbsPair(H);
|
|
else if (IsApplTerm(reg2) && RepAppl(reg2) == myH)
|
|
reg2 = AbsAppl(H);
|
|
}
|
|
*bind_ptr = reg2;
|
|
Bind_Global(&(attv->Value), reg2);
|
|
}
|
|
|
|
void
|
|
Yap_WakeUp(CELL *pt0) {
|
|
CELL d0 = *pt0;
|
|
RESET_VARIABLE(pt0);
|
|
TR--;
|
|
WakeAttVar(pt0, d0);
|
|
}
|
|
|
|
|
|
static void
|
|
mark_attvar(CELL *orig)
|
|
{
|
|
register attvar_record *attv = (attvar_record *)orig;
|
|
Int i;
|
|
|
|
Yap_mark_external_reference(&(attv->Value));
|
|
Yap_mark_external_reference(&(attv->Done));
|
|
for (i = 0; i < NUM_OF_ATTS; i++) {
|
|
Yap_mark_external_reference(attv->Atts+2*i+1);
|
|
}
|
|
}
|
|
|
|
#if FROZEN_STACKS
|
|
static Term
|
|
CurrentTime(void) {
|
|
return(MkIntegerTerm(TR-(tr_fr_ptr)Yap_TrailBase));
|
|
}
|
|
#endif
|
|
|
|
static Term
|
|
InitVarTime(void) {
|
|
#if FROZEN_STACKS
|
|
if (B->cp_tr == TR) {
|
|
/* we run the risk of not making non-determinate bindings before
|
|
the end of the night */
|
|
/* so we just init a TR cell that will not harm anyone */
|
|
Bind((CELL *)(TR+1),AbsAppl(H-1));
|
|
}
|
|
return(MkIntegerTerm(B->cp_tr-(tr_fr_ptr)Yap_TrailBase));
|
|
#else
|
|
Term t = (CELL)H;
|
|
*H++ = TermFoundVar;
|
|
return(t);
|
|
#endif
|
|
}
|
|
|
|
static Int
|
|
PutAtt(attvar_record *attv, Int i, Term tatt) {
|
|
Int pos = i*2;
|
|
#if FROZEN_STACKS
|
|
tr_fr_ptr timestmp = (tr_fr_ptr)Yap_TrailBase+IntegerOfTerm(attv->Atts[pos]);
|
|
if (B->cp_tr <= timestmp
|
|
&& timestmp <= TR) {
|
|
#if defined(SBA)
|
|
if (Unsigned((Int)(attv)-(Int)(H_FZ)) >
|
|
Unsigned((Int)(B_FZ)-(Int)(H_FZ))) {
|
|
CELL *ptr = STACK_TO_SBA(attv->Atts+pos+1);
|
|
*ptr = tatt;
|
|
} else
|
|
#endif
|
|
attv->Atts[pos+1] = tatt;
|
|
if (Unsigned((Int)(attv)-(Int)(HBREG)) >
|
|
Unsigned(BBREG)-(Int)(HBREG))
|
|
TrailVal(timestmp-1) = tatt;
|
|
} else {
|
|
Term tnewt;
|
|
MaBind(attv->Atts+pos+1, tatt);
|
|
tnewt = CurrentTime();
|
|
MaBind(attv->Atts+pos, tnewt);
|
|
}
|
|
#else
|
|
CELL *timestmp = (CELL *)(attv->Atts[pos]);
|
|
if (B->cp_h <= timestmp) {
|
|
attv->Atts[pos+1] = tatt;
|
|
} else {
|
|
Term tnewt;
|
|
MaBind(attv->Atts+pos+1, tatt);
|
|
tnewt = (Term)H;
|
|
*H++ = TermFoundVar;
|
|
MaBind(attv->Atts+pos, tnewt);
|
|
}
|
|
#endif
|
|
return(TRUE);
|
|
}
|
|
|
|
static Int
|
|
UpdateAtt(attvar_record *attv, Int i, Term tatt) {
|
|
Int pos = i*2;
|
|
Term tv = attv->Atts[pos+1];
|
|
if (!IsVarTerm(tv) || !IsUnboundVar(tv)) {
|
|
tatt = MkPairTerm(tatt, attv->Atts[pos+1]);
|
|
} else {
|
|
tatt = MkPairTerm(tatt, TermNil);
|
|
}
|
|
return PutAtt(attv, i, tatt);
|
|
}
|
|
|
|
static Int
|
|
RmAtt(attvar_record *attv, Int i) {
|
|
Int pos = i *2;
|
|
if (!IsVarTerm(attv->Atts[pos+1])) {
|
|
#if FROZEN_STACKS
|
|
tr_fr_ptr timestmp = (tr_fr_ptr)Yap_TrailBase+IntegerOfTerm(attv->Atts[pos]);
|
|
if (B->cp_tr <= timestmp
|
|
&& timestmp <= TR) {
|
|
RESET_VARIABLE(attv->Atts+(pos+1));
|
|
if (Unsigned((Int)(attv)-(Int)(HBREG)) >
|
|
Unsigned(BBREG)-(Int)(HBREG))
|
|
TrailVal(timestmp-1) = attv->Atts[pos+1];
|
|
} else {
|
|
/* reset the variable */
|
|
Term tnewt;
|
|
#ifdef SBA
|
|
MaBind(attv->Atts+(pos+1), 0L);
|
|
#else
|
|
MaBind(attv->Atts+(pos+1), (CELL)(attv->Atts+(pos+1)));
|
|
#endif
|
|
tnewt = CurrentTime();
|
|
MaBind(attv->Atts+pos, tnewt);
|
|
}
|
|
#else
|
|
CELL *timestmp = (CELL *)(attv->Atts[pos]);
|
|
if (B->cp_h <= timestmp) {
|
|
RESET_VARIABLE(attv->Atts+(pos+1));
|
|
} else {
|
|
/* reset the variable */
|
|
Term tnewt;
|
|
#ifdef SBA
|
|
MaBind(attv->Atts+(pos+1), 0L);
|
|
#else
|
|
MaBind(attv->Atts+(pos+1), (CELL)(attv->Atts+(pos+1)));
|
|
#endif
|
|
tnewt = (Term)H;
|
|
*H++ = TermFoundVar;
|
|
MaBind(attv->Atts+pos, tnewt);
|
|
}
|
|
#endif
|
|
}
|
|
return(TRUE);
|
|
}
|
|
|
|
static Int
|
|
BuildNewAttVar(Term t, Int i, Term tatt)
|
|
{
|
|
/* allocate space in Heap */
|
|
Term time;
|
|
int j;
|
|
|
|
attvar_record *attv = (attvar_record *)Yap_ReadTimedVar(DelayedVars);
|
|
if (H0 - (CELL *)attv < 1024+(2*NUM_OF_ATTS)) {
|
|
H[0] = t;
|
|
H[1] = tatt;
|
|
H += 2;
|
|
if (!Yap_growglobal(NULL)) {
|
|
Yap_Error(SYSTEM_ERROR, t, Yap_ErrorMessage);
|
|
return FALSE;
|
|
}
|
|
H -= 2;
|
|
t = H[0];
|
|
tatt = H[1];
|
|
attv = (attvar_record *)Yap_ReadTimedVar(DelayedVars);
|
|
}
|
|
time = InitVarTime();
|
|
RESET_VARIABLE(&(attv->Value));
|
|
RESET_VARIABLE(&(attv->Done));
|
|
attv->sus_id = attvars_ext;
|
|
for (j = 0; j < NUM_OF_ATTS; j++) {
|
|
attv->Atts[2*j] = time;
|
|
RESET_VARIABLE(attv->Atts+(2*j+1));
|
|
}
|
|
attv->NS = Yap_UpdateTimedVar(AttsMutableList, (CELL)&(attv->Done));
|
|
Bind((CELL *)t,(CELL)attv);
|
|
Yap_UpdateTimedVar(DelayedVars,(CELL)(attv->Atts+2*j));
|
|
/* avoid trouble in gc */
|
|
/* if i < 0 then we have the list of arguments */
|
|
if (i < 0) {
|
|
Int j = 0;
|
|
while (IsPairTerm(tatt)) {
|
|
Term t = HeadOfTerm(tatt);
|
|
/* I need to do this because BuildNewAttVar may shift the stacks */
|
|
if (!IsVarTerm(t)) {
|
|
attv->Atts[2*j+1] = t;
|
|
}
|
|
j++;
|
|
tatt = TailOfTerm(tatt);
|
|
}
|
|
return(TRUE);
|
|
} else {
|
|
return(PutAtt(attv, i, tatt));
|
|
}
|
|
}
|
|
|
|
static Int
|
|
GetAtt(attvar_record *attv, int i) {
|
|
Int pos = i *2;
|
|
#if SBA
|
|
if (IsVarTerm(attv->Atts[pos+1]) && IsUnboundVar(attv->Atts[pos+1]))
|
|
return((CELL)&(attv->Atts[pos+1]));
|
|
#endif
|
|
return(attv->Atts[pos+1]);
|
|
}
|
|
|
|
static Int
|
|
FreeAtt(attvar_record *attv, int i) {
|
|
Int pos = i *2;
|
|
return(IsVarTerm(attv->Atts[pos+1]));
|
|
}
|
|
|
|
static Int
|
|
BindAttVar(attvar_record *attv) {
|
|
if (IsVarTerm(attv->Done) && IsUnboundVar(attv->Done)) {
|
|
/* make sure we are not trying to bind a variable against itself */
|
|
if (!IsVarTerm(attv->Value)) {
|
|
Bind_Global(&(attv->Done), attv->Value);
|
|
} else if (IsVarTerm(attv->Value)) {
|
|
Term t = Deref(attv->Value);
|
|
if (IsVarTerm(t)) {
|
|
if (IsAttachedTerm(t)) {
|
|
attvar_record *attv2 = (attvar_record *)VarOfTerm(t);
|
|
if (attv2 < attv) {
|
|
Bind_Global(&(attv->Done), t);
|
|
} else {
|
|
Bind_Global(&(attv2->Done), (CELL)attv);
|
|
}
|
|
} else {
|
|
Yap_Error(SYSTEM_ERROR,(CELL)&(attv->Done),"attvar was bound when unset");
|
|
return(FALSE);
|
|
}
|
|
} else {
|
|
Bind_Global(&(attv->Done), t);
|
|
}
|
|
}
|
|
return(TRUE);
|
|
} else {
|
|
Yap_Error(SYSTEM_ERROR,(CELL)&(attv->Done),"attvar was bound when set");
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
static Term
|
|
GetAllAtts(attvar_record *attv) {
|
|
Int i;
|
|
Term t = TermNil;
|
|
for (i = NUM_OF_ATTS-1; i >= 0; i --) {
|
|
if (!IsVarTerm(attv->Atts[2*i+1]))
|
|
t = MkPairTerm(MkPairTerm(MkIntegerTerm(i),attv->Atts[2*i+1]), t);
|
|
}
|
|
return(t);
|
|
}
|
|
|
|
static Term
|
|
AllAttVars(Term t) {
|
|
if (t == TermNil) {
|
|
return(t);
|
|
} else {
|
|
attvar_record *attv = (attvar_record *)VarOfTerm(t);
|
|
if (!IsVarTerm(attv->Done) || !IsUnboundVar(attv->Done))
|
|
return(AllAttVars(attv->NS));
|
|
else return(MkPairTerm(t,AllAttVars(attv->NS)));
|
|
}
|
|
}
|
|
|
|
Term
|
|
Yap_CurrentAttVars(void) {
|
|
return(AllAttVars(Yap_ReadTimedVar(AttsMutableList)));
|
|
|
|
}
|
|
|
|
static Int
|
|
p_put_att(void) {
|
|
/* receive a variable in ARG1 */
|
|
Term inp = Deref(ARG1);
|
|
/* if this is unbound, ok */
|
|
if (IsVarTerm(inp)) {
|
|
if (IsAttachedTerm(inp)) {
|
|
attvar_record *attv = (attvar_record *)VarOfTerm(inp);
|
|
exts id = (exts)attv->sus_id;
|
|
|
|
if (id != attvars_ext) {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"put_attributes/2");
|
|
return(FALSE);
|
|
}
|
|
return(PutAtt(attv, IntegerOfTerm(Deref(ARG2)), Deref(ARG3)));
|
|
}
|
|
return(BuildNewAttVar(inp, IntegerOfTerm(Deref(ARG2)), Deref(ARG3)));
|
|
} else {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"put_attributes/2");
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
static Int
|
|
p_update_att(void) {
|
|
/* receive a variable in ARG1 */
|
|
Term inp = Deref(ARG1);
|
|
/* if this is unbound, ok */
|
|
if (IsVarTerm(inp)) {
|
|
if (IsAttachedTerm(inp)) {
|
|
attvar_record *attv = (attvar_record *)VarOfTerm(inp);
|
|
exts id = (exts)attv->sus_id;
|
|
|
|
if (id != attvars_ext) {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"put_attributes/2");
|
|
return(FALSE);
|
|
}
|
|
return(UpdateAtt(attv, IntegerOfTerm(Deref(ARG2)), Deref(ARG3)));
|
|
}
|
|
return(BuildNewAttVar(inp, IntegerOfTerm(Deref(ARG2)), MkPairTerm(Deref(ARG3),TermNil)));
|
|
} else {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"put_attributes/2");
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
static Int
|
|
p_rm_att(void) {
|
|
/* receive a variable in ARG1 */
|
|
Term inp = Deref(ARG1);
|
|
/* if this is unbound, ok */
|
|
if (IsVarTerm(inp)) {
|
|
if (IsAttachedTerm(inp)) {
|
|
attvar_record *attv = (attvar_record *)VarOfTerm(inp);
|
|
exts id = (exts)attv->sus_id;
|
|
|
|
if (id != attvars_ext) {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"delete_attribute/2");
|
|
return(FALSE);
|
|
}
|
|
return(RmAtt(attv, IntegerOfTerm(Deref(ARG2))));
|
|
}
|
|
return(TRUE);
|
|
} else {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"delete_attribute/2");
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
static Int
|
|
p_get_att(void) {
|
|
/* receive a variable in ARG1 */
|
|
Term inp = Deref(ARG1);
|
|
/* if this is unbound, ok */
|
|
if (IsVarTerm(inp)) {
|
|
if (IsAttachedTerm(inp)) {
|
|
attvar_record *attv = (attvar_record *)VarOfTerm(inp);
|
|
Term out;
|
|
exts id = (exts)attv->sus_id;
|
|
|
|
if (id != attvars_ext) {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"get_att/2");
|
|
return(FALSE);
|
|
}
|
|
out = GetAtt(attv,IntegerOfTerm(Deref(ARG2)));
|
|
return(!IsVarTerm(out) && Yap_unify(ARG3,out));
|
|
}
|
|
/* Yap_Error(INSTANTIATION_ERROR,inp,"get_att/2");*/
|
|
return(FALSE);
|
|
} else {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"get_att/2");
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
static Int
|
|
p_free_att(void) {
|
|
/* receive a variable in ARG1 */
|
|
Term inp = Deref(ARG1);
|
|
/* if this is unbound, ok */
|
|
if (IsVarTerm(inp)) {
|
|
if (IsAttachedTerm(inp)) {
|
|
attvar_record *attv = (attvar_record *)VarOfTerm(inp);
|
|
exts id = (exts)attv->sus_id;
|
|
|
|
if (id != attvars_ext) {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"get_att/2");
|
|
return(FALSE);
|
|
}
|
|
return(FreeAtt(attv,IntegerOfTerm(Deref(ARG2))));
|
|
}
|
|
return(TRUE);
|
|
} else {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"free_att/2");
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
static Int
|
|
p_bind_attvar(void) {
|
|
/* receive a variable in ARG1 */
|
|
Term inp = Deref(ARG1);
|
|
/* if this is unbound, ok */
|
|
if (IsVarTerm(inp)) {
|
|
if (IsAttachedTerm(inp)) {
|
|
attvar_record *attv = (attvar_record *)VarOfTerm(inp);
|
|
exts id = (exts)attv->sus_id;
|
|
|
|
if (id != attvars_ext) {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"get_att/2");
|
|
return(FALSE);
|
|
}
|
|
return(BindAttVar(attv));
|
|
}
|
|
return(TRUE);
|
|
} else {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"bind_att/2");
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
static Int
|
|
p_get_all_atts(void) {
|
|
/* receive a variable in ARG1 */
|
|
Term inp = Deref(ARG1);
|
|
/* if this is unbound, ok */
|
|
if (IsVarTerm(inp)) {
|
|
if (IsAttachedTerm(inp)) {
|
|
attvar_record *attv = (attvar_record *)VarOfTerm(inp);
|
|
exts id = (exts)(attv->sus_id);
|
|
|
|
if (id != attvars_ext) {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"get_att/2");
|
|
return(FALSE);
|
|
}
|
|
return(Yap_unify(ARG2,GetAllAtts(attv)));
|
|
}
|
|
return(TRUE);
|
|
} else {
|
|
Yap_Error(TYPE_ERROR_VARIABLE,inp,"get_att/2");
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
static Int
|
|
p_inc_atts(void)
|
|
{
|
|
Term t = MkIntegerTerm(NUM_OF_ATTS);
|
|
NUM_OF_ATTS++;
|
|
return(Yap_unify(ARG1,t));
|
|
}
|
|
|
|
static Int
|
|
p_n_atts(void)
|
|
{
|
|
Term t = MkIntegerTerm(NUM_OF_ATTS);
|
|
return Yap_unify(ARG1,t);
|
|
}
|
|
|
|
static Int
|
|
p_all_attvars(void)
|
|
{
|
|
Term t = Yap_ReadTimedVar(AttsMutableList);
|
|
return Yap_unify(ARG1,AllAttVars(t));
|
|
}
|
|
|
|
static Int
|
|
p_is_attvar(void)
|
|
{
|
|
Term t = Deref(ARG1);
|
|
return(IsVarTerm(t) &&
|
|
IsAttachedTerm(t) &&
|
|
((attvar_record *)VarOfTerm(t))->sus_id == attvars_ext);
|
|
}
|
|
|
|
/* check if we are not redoing effort */
|
|
static Int
|
|
p_attvar_bound(void)
|
|
{
|
|
Term t = Deref(ARG1);
|
|
return(IsVarTerm(t) &&
|
|
IsAttachedTerm(t) &&
|
|
((attvar_record *)VarOfTerm(t))->sus_id == attvars_ext &&
|
|
!IsUnboundVar(((attvar_record *)VarOfTerm(t))->Done));
|
|
}
|
|
|
|
#else
|
|
|
|
static Int
|
|
p_all_attvars(void)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static Int
|
|
p_is_attvar(void)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static Int
|
|
p_attvar_bound(void)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
#endif /* COROUTINING */
|
|
|
|
void Yap_InitAttVarPreds(void)
|
|
{
|
|
Term OldCurrentModule = CurrentModule;
|
|
CurrentModule = ATTRIBUTES_MODULE;
|
|
#ifdef COROUTINING
|
|
attas[attvars_ext].bind_op = WakeAttVar;
|
|
attas[attvars_ext].copy_term_op = CopyAttVar;
|
|
attas[attvars_ext].to_term_op = AttVarToTerm;
|
|
attas[attvars_ext].term_to_op = TermToAttVar;
|
|
attas[attvars_ext].mark_op = mark_attvar;
|
|
Yap_InitCPred("get_att", 3, p_get_att, SafePredFlag);
|
|
Yap_InitCPred("get_all_atts", 2, p_get_all_atts, SafePredFlag);
|
|
Yap_InitCPred("free_att", 2, p_free_att, SafePredFlag);
|
|
Yap_InitCPred("put_att", 3, p_put_att, 0);
|
|
Yap_InitCPred("update_att", 3, p_update_att, 0);
|
|
Yap_InitCPred("rm_att", 2, p_rm_att, SafePredFlag);
|
|
Yap_InitCPred("inc_n_of_atts", 1, p_inc_atts, SafePredFlag);
|
|
Yap_InitCPred("n_of_atts", 1, p_n_atts, SafePredFlag);
|
|
Yap_InitCPred("bind_attvar", 1, p_bind_attvar, SafePredFlag);
|
|
#endif /* COROUTINING */
|
|
Yap_InitCPred("all_attvars", 1, p_all_attvars, SafePredFlag);
|
|
CurrentModule = OldCurrentModule;
|
|
Yap_InitCPred("attvar", 1, p_is_attvar, SafePredFlag|TestPredFlag);
|
|
Yap_InitCPred("$att_bound", 1, p_attvar_bound, SafePredFlag|TestPredFlag);
|
|
}
|
|
|
|
|
|
|