/************************************************************************* * * * YAP Prolog * * * * Yap Prolog was developed at NCCUP - Universidade do Porto * * * * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997 * * * ************************************************************************** * * * File: non backtrackable term support * * Last rev: 2/8/06 * * mods: * * comments: non-backtrackable term support * * * *************************************************************************/ #ifdef SCCS static char SccsId[] = "%W% %G%"; #endif /** * @file globals.c * @author VITOR SANTOS COSTA * @date Tue Nov 17 23:16:17 2015 * * @brief support for backtrable and non-backtrackable variables in Prolog. * * */ /** @defgroup Global_Variables Global Variables @ingroup builtins @{ Global variables are associations between names (atoms) and terms. They differ in various ways from storing information using assert/1 or recorda/3. + The value lives on the Prolog (global) stack. This implies that lookup time is independent from the size of the term. This is particularly interesting for large data structures such as parsed XML documents or the CHR global constraint store. + They support both global assignment using nb_setval/2 and backtrackable assignment using b_setval/2. + Only one value (which can be an arbitrary complex Prolog term) can be associated to a variable at a time. + Their value cannot be shared among threads. Each thread has its own namespace and values for global variables. Currently global variables are scoped globally. We may consider module scoping in future versions. Both b_setval/2 and nb_setval/2 implicitly create a variable if the referenced name does not already refer to a variable. Global variables may be initialized from directives to make them available during the program lifetime, but some considerations are necessary for saved-states and threads. Saved-states to not store global variables, which implies they have to be declared with initialization/1 to recreate them after loading the saved state. Each thread has its own set of global variables, starting with an empty set. Using `thread_initialization/1` to define a global variable it will be defined, restored after reloading a saved state and created in all threads that are created after the registration. Finally, global variables can be initialized using the exception hook called exception/3. The latter technique is used by CHR. SWI-Prolog global variables are associations between names (atoms) and terms. They differ in various ways from storing information using assert/1 or recorda/3. + The value lives on the Prolog (global) stack. This implies that lookup time is independent from the size of the term. This is particulary interesting for large data structures qqqsuch as parsed XML documents or the CHR global constraint store. They support both global assignment using nb_setval/2 and backtrackable assignment using b_setval/2. + Only one value (which can be an arbitrary complex Prolog term) can be associated to a variable at a time. + Their value cannot be shared among threads. Each thread has its own namespace and values for global variables. + Currently global variables are scoped globally. We may consider module scoping in future versions. Both b_setval/2 and nb_setval/2 implicitly create a variable if the referenced name does not already refer to a variable. Global variables may be initialized from directives to make them available during the program lifetime, but some considerations are necessary for saved-states and threads. Saved-states to not store global variables, which implies they have to be declared with initialization/1 to recreate them after loading the saved state. Each thread has its own set of global variables, starting with an empty set. Using `thread_inititialization/1` to define a global variable it will be defined, restored after reloading a saved state and created in all threads that are created after the registration. */ #include "Yap.h" #include "Yatom.h" #include "YapHeap.h" #include "yapio.h" #include "iopreds.h" #include "YapEval.h" #include "attvar.h" #include /* Non-backtrackable terms will from now on be stored on arenas, a special term on the heap. Arenas automatically contract as we add terms to the front. */ #define QUEUE_FUNCTOR_ARITY 4 #define QUEUE_ARENA 0 #define QUEUE_HEAD 1 #define QUEUE_TAIL 2 #define QUEUE_SIZE 3 #define HEAP_FUNCTOR_MIN_ARITY #define HEAP_SIZE 0 #define HEAP_MAX 1 #define HEAP_ARENA 2 #define HEAP_START 3 /// A cell_space is a short code region, where we want bindings to proceed locally. /// It is used in copy_term,,,,,,,,,,,,,,,,,,,,,,, /// typedef struct cell_space { struct cell_space *parent; //` CELL *oASP, *oH, *oHB; } cell_space_t; INLINE_ONLY void enter_cell_space(cell_space_t *cs) { cs->oH = HR; cs->oHB = HB; cs->oASP = ASP; } INLINE_ONLY void exit_cell_space(cell_space_t *cs) { HR = cs->oH; HB = cs->oHB; ASP = cs->oASP; } #define MIN_ARENA_SIZE (1048L) #define MAX_ARENA_SIZE (2048 * 16) #define Global_MkIntegerTerm(I) MkIntegerTerm(I) static UInt big2arena_sz(CELL *arena_base) { return (((MP_INT *)(arena_base + 2))->_mp_alloc * sizeof(mp_limb_t) + sizeof(MP_INT) + sizeof(Functor) + 2 * sizeof(CELL)) / sizeof(CELL); } static UInt arena2big_sz(UInt sz) { return sz - (sizeof(MP_INT) + sizeof(Functor) + 2 * sizeof(CELL)) / sizeof(CELL); } /* pointer to top of an arena */ static inline CELL *ArenaLimit(Term arena) { CELL *arena_base = RepAppl(arena); UInt sz = big2arena_sz(arena_base); return arena_base + sz; } /* pointer to top of an arena */ CELL *Yap_ArenaLimit(Term arena) { return ArenaLimit(arena); } /* pointer to top of an arena */ static inline CELL *ArenaPt(Term arena) { return (CELL *)RepAppl(arena); } static inline UInt ArenaSz(Term arena) { return big2arena_sz(RepAppl(arena)); } static Term CreateNewArena(CELL *ptr, UInt size) { Term t = AbsAppl(ptr); MP_INT *dst; ptr[0] = (CELL)FunctorBigInt; ptr[1] = EMPTY_ARENA; dst = (MP_INT *)(ptr + 2); dst->_mp_size = 0L; dst->_mp_alloc = (sizeof(CELL) / sizeof(mp_limb_t)) * arena2big_sz(size); ptr[size - 1] = EndSpecials; return t; } static Term NewArena(UInt size, int wid, UInt arity, CELL *where, struct cell_space *cellSpace) { Term t; UInt new_size; WORKER_REGS(wid) exit_cell_space(cellSpace); if (where == NULL || where == HR) { t = CreateNewArena(HR, size); HR += size; new_size = size; } else { if ((new_size = Yap_InsertInGlobal(where, size * sizeof(CELL))) == 0) { Yap_ThrowError(RESOURCE_ERROR_STACK, TermNil, "No Stack Space for Non-Backtrackable terms"); return TermNil; } size = new_size / sizeof(CELL); t = CreateNewArena(where, size); } enter_cell_space(cellSpace); return t; } static Int p_allocate_arena(USES_REGS1) { cell_space_t cspace; Term t = Deref(ARG1); if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "allocate_arena"); return FALSE; } else if (!IsIntegerTerm(t)) { Yap_Error(TYPE_ERROR_INTEGER, t, "allocate_arena"); return FALSE; } enter_cell_space(&cspace); return Yap_unify(ARG2, NewArena(IntegerOfTerm(t), worker_id, 1, NULL, &cspace)); } static Int p_default_arena_size(USES_REGS1) { return Yap_unify(ARG1, MkIntegerTerm(ArenaSz(LOCAL_GlobalArena))); } void Yap_AllocateDefaultArena(size_t gsize, int wid, void *cs) { cell_space_t ics; if (cs == NULL) { enter_cell_space(&ics); cs = &ics; } REMOTE_GlobalArena(wid) = NewArena(gsize, wid, 2, NULL, cs); } static void adjust_cps(UInt size USES_REGS) { /* adjust possible back pointers in choice-point stack */ choiceptr b_ptr = B; while (b_ptr->cp_h == HR) { b_ptr->cp_h += size; b_ptr = b_ptr->cp_b; } } static Term GrowArena(Term arena, CELL *pt, size_t old_size, size_t size, UInt arity, cell_space_t *cspace USES_REGS) { size_t sz; LOCAL_ArenaOverflows++; if (size == 0) { if (old_size < 128 * 1024) { size = old_size; } else { size = old_size + 128 * 1024; } } if (size < 4*MIN_ARENA_SIZE) { size = 4*MIN_ARENA_SIZE; } while (HR + size > ASP - MIN_ARENA_SIZE) { XREGS[arity + 1] = arena; if (!Yap_gcl(size * sizeof(CELL), arity + 1, ENV, gc_P(P, CP))) { Yap_Error(RESOURCE_ERROR_STACK, TermNil, LOCAL_ErrorMessage); return false; } arena = XREGS[arity + 1]; adjust_cps(size PASS_REGS); } pt = ArenaLimit(arena); sz = old_size + size; if (pt == HR) { HR += size; HR[ - 1] = EndSpecials; } else { if ((sz = Yap_InsertInGlobal(pt, size * sizeof(CELL))) == 0) { return FALSE; } pt[size-1] = EndSpecials; } arena = XREGS[arity + 1]; MP_INT *dst = (MP_INT *)(RepAppl(arena) + 2); dst->_mp_alloc+= size; arena = XREGS[arity + 1]; return sz; } CELL *Yap_GetFromArena(Term *arenap, UInt cells, UInt arity) { CACHE_REGS Term arena = *arenap; CELL *max = ArenaLimit(arena); CELL *base = ArenaPt(arena); CELL *newH; UInt old_sz = ArenaSz(arena), new_size; while(true) { if (IN_BETWEEN(base, HR, max)) { base = HR; HR += cells; return base; } if (base + cells > ASP - 1024) { continue; } newH = base + cells; new_size = old_sz - cells; *arenap = CreateNewArena(newH, new_size); return base; } } static void CloseArena(cell_space_t *region, Term *oldArenaP, UInt old_size USES_REGS) { UInt new_size; if (HR == region->oH) return; new_size = old_size - (HR - RepAppl(*oldArenaP)); *oldArenaP = CreateNewArena(HR, new_size); exit_cell_space( region ); } static inline void clean_dirty_tr(tr_fr_ptr TR0 USES_REGS) { if (TR != TR0) { tr_fr_ptr pt = TR0; do { Term p = TrailTerm(pt++); if (IsVarTerm(p)) { RESET_VARIABLE(p); } else { /* copy downwards */ TrailTerm(TR0 + 1) = TrailTerm(pt); TrailTerm(TR0) = TrailTerm(TR0 + 2) = p; pt += 2; TR0 += 3; } } while (pt != TR); TR = TR0; } } #define expand_stack(S0,SP,SF,TYPE) \ { size_t sz = SF-S0, used = SP-S0; \ S0 = Realloc(S0, (1024+sz)*sizeof(TYPE) PASS_REGS); \ SP = S0+used; SF = S0+(1024+sz); } static int copy_complex_term(register CELL *pt0, register CELL *pt0_end, int share, int copy_att_vars, CELL *ptf, CELL *HLow USES_REGS) { int lvl = push_text_stack(); struct cp_frame *to_visit0, *to_visit = Malloc(1024*sizeof(struct cp_frame)); struct cp_frame *to_visit_max; CELL *HB0 = HB; tr_fr_ptr TR0 = TR; int ground = TRUE; HB = HLow; to_visit0 = to_visit; to_visit_max = to_visit+1024; loop: while (pt0 < pt0_end) { register CELL d0; register CELL *ptd0; ++pt0; ptd0 = pt0; d0 = *ptd0; deref_head(d0, copy_term_unk); copy_term_nvar : { if (IsPairTerm(d0)) { CELL *ap2 = RepPair(d0); if ((share && ap2 < HB) || (ap2 >= HB && ap2 < HR)) { /* If this is newer than the current term, just reuse */ *ptf++ = d0; continue; } *ptf = AbsPair(HR); ptf++; #ifdef RATIONAL_TREES if (to_visit >= to_visit_max-32) { expand_stack(to_visit0, to_visit, to_visit_max, struct cp_frame); } to_visit->start_cp = pt0; to_visit->end_cp = pt0_end; to_visit->to = ptf; to_visit->oldv = *pt0; to_visit->ground = ground; /* fool the system into thinking we had a variable there */ *pt0 = AbsPair(HR); to_visit++; #else if (pt0 < pt0_end) { if (to_visit + 32 >= to_visit_max - 32) { expand_stack(to_visit0, to_visit, to_visit_max, struct cp_frame); } to_visit->start_cp = pt0; to_visit->end_cp = pt0_end; to_visit->to = ptf; to_visit->ground = ground; to_visit++; } #endif ground = TRUE; pt0 = ap2 - 1; pt0_end = ap2 + 1; ptf = HR; HR += 2; if (HR > ASP - MIN_ARENA_SIZE) { goto overflow; } } else if (IsApplTerm(d0)) { register Functor f; register CELL *ap2; /* store the terms to visit */ ap2 = RepAppl(d0); if ((share && ap2 < HB) || (ap2 >= HB && ap2 < HR)) { /* If this is newer than the current term, just reuse */ *ptf++ = d0; continue; } f = (Functor)(*ap2); if (IsExtensionFunctor(f)) { switch ((CELL)f) { case (CELL) FunctorDBRef: case (CELL) FunctorAttVar: *ptf++ = d0; break; case (CELL) FunctorLongInt: if (HR > ASP - (MIN_ARENA_SIZE + 3)) { goto overflow; } *ptf++ = AbsAppl(HR); HR[0] = (CELL)f; HR[1] = ap2[1]; HR[2] = EndSpecials; HR += 3; if (HR > ASP - MIN_ARENA_SIZE) { goto overflow; } break; case (CELL) FunctorDouble: if (HR > ASP - (MIN_ARENA_SIZE + (2 + SIZEOF_DOUBLE / sizeof(CELL)))) { goto overflow; } *ptf++ = AbsAppl(HR); HR[0] = (CELL)f; HR[1] = ap2[1]; #if SIZEOF_DOUBLE == 2 * SIZEOF_INT_P HR[2] = ap2[2]; HR[3] = EndSpecials; HR += 4; #else HR[2] = EndSpecials; HR += 3; #endif break; case (CELL) FunctorString: if (ASP - HR < MIN_ARENA_SIZE + 3 + ap2[1]) { goto overflow; } *ptf++ = AbsAppl(HR); memmove(HR, ap2, sizeof(CELL) * (3 + ap2[1])); HR += ap2[1] + 3; break; default: { /* big int */ UInt sz = (sizeof(MP_INT) + 3 * CellSize + ((MP_INT *)(ap2 + 2))->_mp_alloc * sizeof(mp_limb_t)) / CellSize, i; if (HR > ASP - (MIN_ARENA_SIZE + sz)) { goto overflow; } *ptf++ = AbsAppl(HR); HR[0] = (CELL)f; for (i = 1; i < sz; i++) { HR[i] = ap2[i]; } HR += sz; } } continue; } *ptf = AbsAppl(HR); ptf++; /* store the terms to visit */ #ifdef RATIONAL_TREES if (to_visit + 32 >= to_visit_max) { expand_stack(to_visit0, to_visit, to_visit_max, struct cp_frame); } to_visit->start_cp = pt0; to_visit->end_cp = pt0_end; to_visit->to = ptf; to_visit->oldv = *pt0; to_visit->ground = ground; /* fool the system into thinking we had a variable there */ *pt0 = AbsAppl(HR); to_visit++; #else if (pt0 < pt0_end) { if (to_visit++ >= (CELL **)AuxSp) { expand_stack(to_visit0, to_visit, to_visit_max, struct cp_frame); } to_visit->start_cp = pt0; to_visit->end_cp = pt0_end; to_visit->to = ptf; to_visit->ground = ground; to_visit++; } #endif ground = (f != FunctorMutable); d0 = ArityOfFunctor(f); pt0 = ap2; pt0_end = ap2 + d0; /* store the functor for the new term */ HR[0] = (CELL)f; ptf = HR + 1; HR += 1 + d0; if (HR > ASP - MIN_ARENA_SIZE) { goto overflow; } } else { /* just copy atoms or integers */ *ptf++ = d0; } continue; } derefa_body(d0, ptd0, copy_term_unk, copy_term_nvar); ground = FALSE; /* don't need to copy variables if we want to share the global term */ if ((share && ptd0 < HB && ptd0 > H0) || (ptd0 >= HLow && ptd0 < HR)) { /* we have already found this cell */ *ptf++ = (CELL)ptd0; } else { #if COROUTINING if (copy_att_vars && GlobalIsAttachedTerm((CELL)ptd0)) { /* if unbound, call the standard copy term routine */ struct cp_frame *bp; CELL new; bp = to_visit; if (!GLOBAL_attas[ExtFromCell(ptd0)].copy_term_op(ptd0, &bp, ptf PASS_REGS)) { goto overflow; } to_visit = bp; new = *ptf; if (TR > (tr_fr_ptr)LOCAL_TrailTop - 256) { /* Trail overflow */ if (!Yap_growtrail((TR - TR0) * sizeof(tr_fr_ptr *), TRUE)) { goto trail_overflow; } } Bind_and_Trail(ptd0, new); ptf++; } else { #endif /* first time we met this term */ RESET_VARIABLE(ptf); if ((ADDR)TR > LOCAL_TrailTop - MIN_ARENA_SIZE) goto trail_overflow; Bind_and_Trail(ptd0, (CELL)ptf); ptf++; #ifdef COROUTINING } #endif } } /* Do we still have compound terms to visit */ if (to_visit > to_visit0) { to_visit--; pt0 = to_visit->start_cp; pt0_end = to_visit->end_cp; ptf = to_visit->to; #ifdef RATIONAL_TREES *pt0 = to_visit->oldv; #endif ground = (ground && to_visit->ground); goto loop; } /* restore our nice, friendly, term to its original state */ HB = HB0; clean_dirty_tr(TR0 PASS_REGS); /* follow chain of multi-assigned variables */ pop_text_stack(lvl); return 0; overflow: /* oops, we're in trouble */ HR = HLow; /* we've done it */ /* restore our nice, friendly, term to its original state */ HB = HB0; #ifdef RATIONAL_TREES while (to_visit > to_visit0) { to_visit--; pt0 = to_visit->start_cp; pt0_end = to_visit->end_cp; ptf = to_visit->to; *pt0 = to_visit->oldv; } #endif reset_trail(TR0); pop_text_stack(lvl); return -1; trail_overflow: /* oops, we're in trouble */ HR = HLow; /* we've done it */ /* restore our nice, friendly, term to its original state */ HB = HB0; #ifdef RATIONAL_TREES while (to_visit > to_visit0) { to_visit--; pt0 = to_visit->start_cp; pt0_end = to_visit->end_cp; ptf = to_visit->to; *pt0 = to_visit->oldv; } #endif reset_trail(TR0); pop_text_stack(lvl); return -4; } static Term CopyTermToArena(Term t, Term arena, bool share, bool copy_att_vars, UInt arity, Term *newarena, size_t min_grow USES_REGS) { size_t old_size = ArenaSz(arena); cell_space_t cspace; int res = 0; Term tn; int restarts = 0; restart: t = Deref(t); enter_cell_space( & cspace); if (IsVarTerm(t)) { ASP = ArenaLimit(arena); HR = HB = ArenaPt(arena); #if COROUTINING if (GlobalIsAttachedTerm(t)) { CELL *Hi; *HR = t; Hi = HR + 1; HR += 2; if ((res = copy_complex_term(Hi - 2, Hi - 1, share, copy_att_vars, Hi, Hi PASS_REGS)) < 0) goto error_handler; CloseArena(&cspace, newarena, old_size PASS_REGS); return Hi[0]; } #endif if (share && VarOfTerm(t) > ArenaPt(arena)) { CloseArena(&cspace, newarena, old_size PASS_REGS); return t; } tn = MkVarTerm(); if (HR > ASP - MIN_ARENA_SIZE) { res = -1; goto error_handler; } CloseArena(&cspace, newarena, old_size PASS_REGS); return tn; } else if (IsAtomOrIntTerm(t)) { return t; } else if (IsPairTerm(t)) { Term tf; CELL *ap; CELL *Hi; if (share && ArenaPt(arena) > RepPair(t)) { return t; } HR = HB = ArenaPt(arena); ASP = ArenaLimit(arena); ap = RepPair(t); Hi = HR; tf = AbsPair(HR); HR += 2; if ((res = copy_complex_term(ap - 1, ap + 1, share, copy_att_vars, Hi, Hi PASS_REGS)) < 0) { goto error_handler; } CloseArena(&cspace, newarena, old_size PASS_REGS); return tf; } else { Functor f; Term tf; CELL *HB0; CELL *ap; if (share && ArenaPt(arena) > RepAppl(t)) { return t; } HR = HB = ArenaPt(arena); ASP = ArenaLimit(arena); f = FunctorOfTerm(t); HB0 = HR; ap = RepAppl(t); tf = AbsAppl(HR); HR[0] = (CELL)f; if (IsExtensionFunctor(f)) { switch ((CELL)f) { case (CELL) FunctorDBRef: CloseArena(&cspace, newarena, old_size PASS_REGS); return t; case (CELL) FunctorLongInt: if (HR > ASP - (MIN_ARENA_SIZE + 3)) { res = -1; goto error_handler; } HR[1] = ap[1]; HR[2] = EndSpecials; HR += 3; break; case (CELL) FunctorDouble: if (HR > ASP - (MIN_ARENA_SIZE + (2 + SIZEOF_DOUBLE / sizeof(CELL)))) { res = -1; goto error_handler; } HR[1] = ap[1]; #if SIZEOF_DOUBLE == 2 * SIZEOF_INT_P HR[2] = ap[2]; HR[3] = EndSpecials; HR += 4; #else HR[2] = EndSpecials; HR += 3; #endif break; case (CELL) FunctorString: if (HR > ASP - (MIN_ARENA_SIZE + 3 + ap[1])) { res = -1; goto error_handler; } memmove(HR, ap, sizeof(CELL) * (3 + ap[1])); HR += ap[1] + 3; break; default: { UInt sz = ArenaSz(t), i; if (HR > ASP - (MIN_ARENA_SIZE + sz)) { res = -1; goto error_handler; } for (i = 1; i < sz; i++) { HR[i] = ap[i]; } HR += sz; } } } else { HR += 1 + ArityOfFunctor(f); if (HR > ASP - MIN_ARENA_SIZE) { res = -1; goto error_handler; } if ((res = copy_complex_term(ap, ap + ArityOfFunctor(f), share, copy_att_vars, HB0 + 1, HB0 PASS_REGS)) < 0) { goto error_handler; } } CloseArena(&cspace, newarena, old_size PASS_REGS); return tf; } error_handler: HR = HB; CloseArena(&cspace, newarena, old_size PASS_REGS); XREGS[arity + 1] = t; XREGS[arity + 2] = arena; XREGS[arity + 3] = (CELL)newarena; { CELL *old_top = ArenaLimit(*newarena); exit_cell_space(&cspace); switch (res) { case -1: if (arena == LOCAL_GlobalArena) LOCAL_GlobalArenaOverflows++; restarts++; min_grow += (restarts < 16 ? 16*1024*restarts*restarts : 128*1024*1024); if ((arena=GrowArena(arena, old_top, old_size, min_grow, arity + 3, &cspace PASS_REGS))==0) { Yap_Error(RESOURCE_ERROR_STACK, TermNil, LOCAL_ErrorMessage); return 0L; } break; default: /* temporary space overflow */ return 0; } } enter_cell_space(&cspace); newarena = (CELL *)XREGS[arity + 3]; arena = Deref(XREGS[arity + 2]); t = XREGS[arity + 1]; old_size = ArenaSz(arena); goto restart; } static Term CreateTermInArena(Term arena, Atom Na, UInt Nar, UInt arity, Term *newarena, Term init USES_REGS) { cell_space_t cells; UInt old_size = ArenaSz(arena); enter_cell_space(&cells); Term tf; CELL *HB0; Functor f = Yap_MkFunctor(Na, Nar); UInt i; restart: HR = HB = ArenaPt(arena); ASP = ArenaLimit(arena); HB0 = HR; tf = AbsAppl(HR); HR[0] = (CELL)f; HR += 1 + ArityOfFunctor(f); if (HR > ASP - MIN_ARENA_SIZE) { /* overflow */ HR = HB; CloseArena(&cells, newarena, old_size PASS_REGS); XREGS[arity + 1] = arena; XREGS[arity + 2] = (CELL)newarena; { CELL *old_top = ArenaLimit(*newarena); if (arena == LOCAL_GlobalArena) LOCAL_GlobalArenaOverflows++; if ((arena=GrowArena(arena, old_top, old_size, Nar * sizeof(CELL), arity + 2, &cells PASS_REGS))==0) { Yap_Error(RESOURCE_ERROR_STACK, TermNil, "while creating large global term"); return 0L; } } newarena = (CELL *)XREGS[arity + 2]; arena = Deref(XREGS[arity + 1]); old_size = ArenaSz(arena); goto restart; } if (init == 0L) { for (i = 1; i <= Nar; i++) { RESET_VARIABLE(HB0 + i); } } else { for (i = 1; i <= Nar; i++) { HB0[i] = init; } } CloseArena(&cells, newarena, old_size PASS_REGS); return tf; } inline static GlobalEntry *FindGlobalEntry(Atom at USES_REGS) /* get predicate entry for ap/arity; create it if neccessary. */ { Prop p0; AtomEntry *ae = RepAtom(at); READ_LOCK(ae->ARWLock); p0 = ae->PropsOfAE; while (p0) { GlobalEntry *pe = RepGlobalProp(p0); if (pe->KindOfPE == GlobalProperty #if THREADS && pe->owner_id == worker_id #endif ) { READ_UNLOCK(ae->ARWLock); return pe; } p0 = pe->NextOfPE; } READ_UNLOCK(ae->ARWLock); return NULL; } inline static GlobalEntry *GetGlobalEntry(Atom at USES_REGS) /* get predicate entry for ap/arity; create it if neccessary. */ { Prop p0; AtomEntry *ae = RepAtom(at); GlobalEntry *new; WRITE_LOCK(ae->ARWLock); p0 = ae->PropsOfAE; while (p0) { GlobalEntry *pe = RepGlobalProp(p0); if (pe->KindOfPE == GlobalProperty #if THREADS && pe->owner_id == worker_id #endif ) { WRITE_UNLOCK(ae->ARWLock); return pe; } p0 = pe->NextOfPE; } new = (GlobalEntry *)Yap_AllocAtomSpace(sizeof(*new)); INIT_RWLOCK(new->GRWLock); new->KindOfPE = GlobalProperty; #if THREADS new->owner_id = worker_id; #endif new->NextGE = LOCAL_GlobalVariables; LOCAL_GlobalVariables = new; new->AtomOfGE = ae; AddPropToAtom(ae, (PropEntry *)new); RESET_VARIABLE(&new->global); WRITE_UNLOCK(ae->ARWLock); return new; } static UInt garena_overflow_size(CELL *arena USES_REGS) { UInt dup = (((CELL *)arena - H0) * sizeof(CELL)) >> 3; if (dup < 64 * 1024 * LOCAL_GlobalArenaOverflows) dup = 64 * 1024 * LOCAL_GlobalArenaOverflows; if (dup > 1024 * 1024) return 1024 * 1024; return dup; } static Int p_nb_setarg(USES_REGS1) { Term wheret = Deref(ARG1); Term dest; Term to; UInt arity, pos; CELL *destp; if (IsVarTerm(wheret)) { Yap_Error(INSTANTIATION_ERROR, wheret, "nb_setarg"); return FALSE; } if (!IsIntegerTerm(wheret)) { Yap_Error(TYPE_ERROR_INTEGER, wheret, "nb_setarg"); return FALSE; } pos = IntegerOfTerm(wheret); dest = Deref(ARG2); if (IsVarTerm(dest)) { Yap_Error(INSTANTIATION_ERROR, dest, "nb_setarg"); return FALSE; } else if (IsPrimitiveTerm(dest)) { arity = 0; } else if (IsPairTerm(dest)) { arity = 2; } else { arity = ArityOfFunctor(FunctorOfTerm(dest)); } if (pos < 1 || pos > arity) return FALSE; to = Deref(ARG3); to = CopyTermToArena( ARG3, LOCAL_GlobalArena, FALSE, TRUE, 3, &LOCAL_GlobalArena, garena_overflow_size(ArenaPt(LOCAL_GlobalArena) PASS_REGS) PASS_REGS); if (to == 0L) return FALSE; dest = Deref(ARG2); if (IsPairTerm(dest)) { destp = RepPair(dest) - 1; } else { destp = RepAppl(dest); } destp[pos] = to; return TRUE; } static Int p_nb_set_shared_arg(USES_REGS1) { Term wheret = Deref(ARG1); Term dest = Deref(ARG2); Term to; UInt arity, pos; CELL *destp; if (IsVarTerm(wheret)) { Yap_Error(INSTANTIATION_ERROR, wheret, "nb_setarg"); return FALSE; } if (!IsIntegerTerm(wheret)) { Yap_Error(TYPE_ERROR_INTEGER, wheret, "nb_setarg"); return FALSE; } pos = IntegerOfTerm(wheret); if (IsVarTerm(dest)) { Yap_Error(INSTANTIATION_ERROR, dest, "nb_setarg"); return FALSE; } else if (IsPrimitiveTerm(dest)) { arity = 0; } else if (IsPairTerm(dest)) { arity = 2; } else { arity = ArityOfFunctor(FunctorOfTerm(dest)); } if (pos < 1 || pos > arity) return FALSE; to = CopyTermToArena( ARG3, LOCAL_GlobalArena, TRUE, TRUE, 3, &LOCAL_GlobalArena, garena_overflow_size(ArenaPt(LOCAL_GlobalArena) PASS_REGS) PASS_REGS); if (to == 0L) return FALSE; if (IsPairTerm(dest)) { destp = RepPair(dest) - 1; } else { destp = RepAppl(dest); } destp[pos] = to; return TRUE; } static Int p_nb_linkarg(USES_REGS1) { Term wheret = Deref(ARG1); Term dest = Deref(ARG2); UInt arity, pos; CELL *destp; if (IsVarTerm(wheret)) { Yap_Error(INSTANTIATION_ERROR, wheret, "nb_setarg"); return FALSE; } if (!IsIntegerTerm(wheret)) { Yap_Error(TYPE_ERROR_INTEGER, wheret, "nb_setarg"); return FALSE; } pos = IntegerOfTerm(wheret); if (IsVarTerm(dest)) { Yap_Error(INSTANTIATION_ERROR, dest, "nb_setarg"); return FALSE; } else if (IsPrimitiveTerm(dest)) { arity = 0; destp = NULL; } else if (IsPairTerm(dest)) { arity = 2; destp = RepPair(dest) - 1; } else { arity = ArityOfFunctor(FunctorOfTerm(dest)); destp = RepAppl(dest); } if (pos < 1 || pos > arity) return FALSE; destp[pos] = Deref(ARG3); return TRUE; } static Int p_nb_linkval(USES_REGS1) { Term t = Deref(ARG1), to; GlobalEntry *ge; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_linkval"); return (TermNil); } else if (!IsAtomTerm(t)) { Yap_Error(TYPE_ERROR_ATOM, t, "nb_linkval"); return (FALSE); } ge = GetGlobalEntry(AtomOfTerm(t) PASS_REGS); to = Deref(ARG2); WRITE_LOCK(ge->GRWLock); ge->global = to; WRITE_UNLOCK(ge->GRWLock); return TRUE; } static Int p_nb_create_accumulator(USES_REGS1) { Term t = Deref(ARG1), acct, to, t2; CELL *destp; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_create_accumulator"); return FALSE; } if (!IsIntegerTerm(t) && !IsBigIntTerm(t) && !IsFloatTerm(t)) { Yap_Error(TYPE_ERROR_NUMBER, t, "nb_create_accumulator"); return FALSE; } acct = Yap_MkApplTerm(FunctorGNumber, 1, &t); if (!Yap_unify(ARG2, acct)) { return FALSE; } to = CopyTermToArena( t, LOCAL_GlobalArena, TRUE, TRUE, 2, &LOCAL_GlobalArena, garena_overflow_size(ArenaPt(LOCAL_GlobalArena) PASS_REGS) PASS_REGS); if (to == 0L) return FALSE; t2 = Deref(ARG2); if (IsVarTerm(t2)) { return Yap_unify(t2, Yap_MkApplTerm(FunctorGNumber, 1, &to)); } destp = RepAppl(Deref(ARG2)); destp[1] = to; return TRUE; } static Int p_nb_add_to_accumulator(USES_REGS1) { Term t = Deref(ARG1), t0, tadd; Functor f; CELL *destp; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_create_accumulator"); return FALSE; } if (!IsApplTerm(t)) { Yap_Error(TYPE_ERROR_NUMBER, t, "nb_accumulator_value"); return FALSE; } f = FunctorOfTerm(t); if (f != FunctorGNumber) { return FALSE; } destp = RepAppl(t); t0 = Deref(destp[1]); tadd = Deref(ARG2); if (IsVarTerm(tadd)) { Yap_Error(INSTANTIATION_ERROR, tadd, "nb_create_accumulator"); return FALSE; } if (IsIntegerTerm(t0) && IsIntegerTerm(tadd)) { Int i0 = IntegerOfTerm(t0); Int i1 = IntegerOfTerm(tadd); Term new = MkIntegerTerm(i0 + i1); if (IsIntTerm(new)) { /* forget it if it was something else */ destp[1] = new; } else { /* long, do we have space or not ?? */ if (IsLongIntTerm(t0)) { CELL *target = RepAppl(t0); CELL *source = RepAppl(new); target[1] = source[1]; } else { /* we need to create a new long int */ new = CopyTermToArena( new, LOCAL_GlobalArena, TRUE, TRUE, 2, &LOCAL_GlobalArena, garena_overflow_size(ArenaPt(LOCAL_GlobalArena) PASS_REGS) PASS_REGS); destp = RepAppl(Deref(ARG1)); destp[1] = new; } } return TRUE; } if (IsFloatTerm(t0) && IsFloatTerm(tadd)) { Float f0 = FloatOfTerm(t0); Float f1 = FloatOfTerm(tadd); Term new = MkFloatTerm(f0 + f1); CELL *target = RepAppl(t0); CELL *source = RepAppl(new); #if SIZEOF_DOUBLE == 2 * SIZEOF_INT_P target[2] = source[2]; #endif target[1] = source[1]; return TRUE; } if (IsNumTerm(t0) && IsNumTerm(tadd)) { Term t2[2], new; t2[0] = t0; t2[1] = tadd; new = Yap_MkApplTerm(FunctorPlus, 2, t2); new = Yap_Eval(new); new = CopyTermToArena( new, LOCAL_GlobalArena, TRUE, TRUE, 2, &LOCAL_GlobalArena, garena_overflow_size(ArenaPt(LOCAL_GlobalArena) PASS_REGS) PASS_REGS); destp = RepAppl(Deref(ARG1)); destp[1] = new; return TRUE; } return FALSE; } static Int p_nb_accumulator_value(USES_REGS1) { Term t = Deref(ARG1); Functor f; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_accumulator_value"); return FALSE; } if (!IsApplTerm(t)) { Yap_Error(TYPE_ERROR_NUMBER, t, "nb_accumulator_value"); return FALSE; } f = FunctorOfTerm(t); if (f != FunctorGNumber) { return FALSE; } return Yap_unify(ArgOfTerm(1, t), ARG2); } Term Yap_SetGlobalVal(Atom at, Term t0) { CACHE_REGS Term to; GlobalEntry *ge; ge = GetGlobalEntry(at PASS_REGS); to = CopyTermToArena( t0, LOCAL_GlobalArena, FALSE, TRUE, 2, &LOCAL_GlobalArena, garena_overflow_size(ArenaPt(LOCAL_GlobalArena) PASS_REGS) PASS_REGS); if (to == 0L) return to; WRITE_LOCK(ge->GRWLock); ge->global = to; WRITE_UNLOCK(ge->GRWLock); return to; } Term Yap_SaveTerm(Term t0) { CACHE_REGS Term to; to = CopyTermToArena( Deref(t0), LOCAL_GlobalArena, FALSE, TRUE, 2, &LOCAL_GlobalArena, garena_overflow_size(ArenaPt(LOCAL_GlobalArena) PASS_REGS) PASS_REGS); if (to == 0L) return to; return to; } static Int p_nb_setval(USES_REGS1) { Term t = Deref(ARG1); if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_setval"); return (TermNil); } else if (!IsAtomTerm(t)) { Yap_Error(TYPE_ERROR_ATOM, t, "nb_setval"); return (FALSE); } return Yap_SetGlobalVal(AtomOfTerm(t), ARG2); } static Int p_nb_set_shared_val(USES_REGS1) { Term t = Deref(ARG1), to; GlobalEntry *ge; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_setval"); return (TermNil); } else if (!IsAtomTerm(t)) { Yap_Error(TYPE_ERROR_ATOM, t, "nb_setval"); return (FALSE); } ge = GetGlobalEntry(AtomOfTerm(t) PASS_REGS); to = CopyTermToArena( ARG2, LOCAL_GlobalArena, TRUE, TRUE, 2, &LOCAL_GlobalArena, garena_overflow_size(ArenaPt(LOCAL_GlobalArena) PASS_REGS) PASS_REGS); if (to == 0L) return FALSE; WRITE_LOCK(ge->GRWLock); ge->global = to; WRITE_UNLOCK(ge->GRWLock); return TRUE; } static Int p_b_setval(USES_REGS1) { Term t = Deref(ARG1); GlobalEntry *ge; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "b_setval"); return (TermNil); } else if (!IsAtomTerm(t)) { Yap_Error(TYPE_ERROR_ATOM, t, "b_setval"); return (FALSE); } ge = GetGlobalEntry(AtomOfTerm(t) PASS_REGS); WRITE_LOCK(ge->GRWLock); #ifdef MULTI_ASSIGNMENT_VARIABLES /* the evil deed is to be done now */ { /* but first make sure we are doing on a global object, or a constant! */ Term t = Deref(ARG2); if (IsVarTerm(t) && VarOfTerm(t) > HR && VarOfTerm(t) < LCL0) { Term tn = MkVarTerm(); Bind_Local(VarOfTerm(t), tn); t = tn; } MaBind(&ge->global, t); } WRITE_UNLOCK(ge->GRWLock); return TRUE; #else WRITE_UNLOCK(ge->GRWLock); Yap_Error(SYSTEM_ERROR_INTERNAL, t, "update_array"); return FALSE; #endif } static int undefined_global(USES_REGS1) { Term t3 = Deref(ARG3); if (IsApplTerm(t3)) { if (FunctorOfTerm(t3) == FunctorEq) return Yap_unify(ArgOfTerm(1, t3), ArgOfTerm(2, t3)); return FALSE; } return Yap_unify(t3, TermNil); } static Int p_nb_getval(USES_REGS1) { Term t = Deref(ARG1), to; GlobalEntry *ge; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_getval"); return FALSE; } else if (!IsAtomTerm(t)) { Yap_Error(TYPE_ERROR_ATOM, t, "nb_getval"); return FALSE; } ge = FindGlobalEntry(AtomOfTerm(t) PASS_REGS); if (!ge) return undefined_global(PASS_REGS1); READ_LOCK(ge->GRWLock); to = ge->global; if (IsVarTerm(to) && IsUnboundVar(VarOfTerm(to))) { Term t = MkVarTerm(); YapBind(VarOfTerm(to), t); to = t; } READ_UNLOCK(ge->GRWLock); if (to == TermFoundVar) { return FALSE; } return Yap_unify(ARG2, to); } Term Yap_GetGlobal(Atom at) { CACHE_REGS GlobalEntry *ge; Term to; ge = FindGlobalEntry(at PASS_REGS); if (!ge) return 0L; READ_LOCK(ge->GRWLock); to = ge->global; if (IsVarTerm(to) && IsUnboundVar(VarOfTerm(to))) { Term t = MkVarTerm(); YapBind(VarOfTerm(to), t); to = t; } READ_UNLOCK(ge->GRWLock); if (to == TermFoundVar) { return 0; } return to; } static Int nbdelete(Atom at USES_REGS) { GlobalEntry *ge, *g; AtomEntry *ae; Prop gp, g0; ge = FindGlobalEntry(at PASS_REGS); if (!ge) { Yap_Error(EXISTENCE_ERROR_VARIABLE, MkAtomTerm(at), "nb_delete"); return FALSE; } WRITE_LOCK(ge->GRWLock); ae = ge->AtomOfGE; if (LOCAL_GlobalVariables == ge) { LOCAL_GlobalVariables = ge->NextGE; } else { g = LOCAL_GlobalVariables; while (g->NextGE != ge) g = g->NextGE; g->NextGE = ge->NextGE; } gp = AbsGlobalProp(ge); WRITE_LOCK(ae->ARWLock); if (ae->PropsOfAE == gp) { ae->PropsOfAE = ge->NextOfPE; } else { g0 = ae->PropsOfAE; while (g0->NextOfPE != gp) g0 = g0->NextOfPE; g0->NextOfPE = ge->NextOfPE; } WRITE_UNLOCK(ae->ARWLock); WRITE_UNLOCK(ge->GRWLock); Yap_FreeCodeSpace((char *)ge); return TRUE; } Int Yap_DeleteGlobal(Atom at) { CACHE_REGS return nbdelete(at PASS_REGS); } static Int p_nb_delete(USES_REGS1) { Term t = Deref(ARG1); if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_delete"); return FALSE; } else if (!IsAtomTerm(t)) { Yap_Error(TYPE_ERROR_ATOM, t, "nb_delete"); return FALSE; } return nbdelete(AtomOfTerm(t) PASS_REGS); } static Int p_nb_create(USES_REGS1) { Term t = Deref(ARG1); Term tname = Deref(ARG2); Term tarity = Deref(ARG3); Term to; GlobalEntry *ge; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_create"); return FALSE; } else if (!IsAtomTerm(t)) { Yap_Error(TYPE_ERROR_ATOM, t, "nb_create"); return FALSE; } ge = GetGlobalEntry(AtomOfTerm(t) PASS_REGS); if (!ge) { Yap_Error(EXISTENCE_ERROR_VARIABLE, t, "nb_create"); return FALSE; } if (IsVarTerm(tarity)) { Yap_Error(INSTANTIATION_ERROR, tarity, "nb_create"); return FALSE; } else if (!IsIntegerTerm(tarity)) { Yap_Error(TYPE_ERROR_INTEGER, tarity, "nb_create"); return FALSE; } if (IsVarTerm(tname)) { Yap_Error(INSTANTIATION_ERROR, tname, "nb_create"); return FALSE; } else if (!IsAtomTerm(tname)) { Yap_Error(TYPE_ERROR_ATOM, tname, "nb_create"); return FALSE; } to = CreateTermInArena(LOCAL_GlobalArena, AtomOfTerm(tname), IntegerOfTerm(tarity), 3, &LOCAL_GlobalArena, 0L PASS_REGS); if (!to) return FALSE; WRITE_LOCK(ge->GRWLock); ge->global = to; WRITE_UNLOCK(ge->GRWLock); return TRUE; } static Int p_nb_create2(USES_REGS1) { Term t = Deref(ARG1); Term tname = Deref(ARG2); Term tarity = Deref(ARG3); Term tinit = Deref(ARG4); Term to; GlobalEntry *ge; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_create"); return FALSE; } else if (!IsAtomTerm(t)) { Yap_Error(TYPE_ERROR_ATOM, t, "nb_create"); return FALSE; } ge = GetGlobalEntry(AtomOfTerm(t) PASS_REGS); if (!ge) { Yap_Error(EXISTENCE_ERROR_VARIABLE, t, "nb_create"); return FALSE; } if (IsVarTerm(tarity)) { Yap_Error(INSTANTIATION_ERROR, tarity, "nb_create"); return FALSE; } else if (!IsIntegerTerm(tarity)) { Yap_Error(TYPE_ERROR_INTEGER, tarity, "nb_create"); return FALSE; } if (IsVarTerm(tname)) { Yap_Error(INSTANTIATION_ERROR, tname, "nb_create"); return FALSE; } else if (!IsAtomTerm(tname)) { Yap_Error(TYPE_ERROR_ATOM, tname, "nb_create"); return FALSE; } if (IsVarTerm(tinit)) { Yap_Error(INSTANTIATION_ERROR, tname, "nb_create"); return FALSE; } else if (!IsAtomTerm(tinit)) { Yap_Error(TYPE_ERROR_ATOM, tname, "nb_create"); return FALSE; } to = CreateTermInArena(LOCAL_GlobalArena, AtomOfTerm(tname), IntegerOfTerm(tarity), 4, &LOCAL_GlobalArena, tinit PASS_REGS); if (!to) return FALSE; WRITE_LOCK(ge->GRWLock); ge->global = to; WRITE_UNLOCK(ge->GRWLock); return TRUE; } /// @{ /// @addtogroup nb /* a non-backtrackable queue is a term of the form $array(Arena,Start,End,Size) * plus an Arena. */ static Int nb_queue(UInt arena_sz USES_REGS) { cell_space_t cspace; Term queue_arena, queue, ar[QUEUE_FUNCTOR_ARITY], *nar; Term t = Deref(ARG1); LOCAL_DepthArenas++; if (!IsVarTerm(t)) { if (!IsApplTerm(t)) { return FALSE; } return (FunctorOfTerm(t) == FunctorNBQueue); } ar[QUEUE_ARENA] = ar[QUEUE_HEAD] = ar[QUEUE_TAIL] = ar[QUEUE_SIZE] = MkIntTerm(0); queue = Yap_MkApplTerm(FunctorNBQueue, QUEUE_FUNCTOR_ARITY, ar); if (!Yap_unify(queue, ARG1)) return FALSE; if (arena_sz < 32 * 1024) arena_sz = 32 * 1024; enter_cell_space(&cspace); queue_arena = NewArena(arena_sz, worker_id, 1, NULL, &cspace); if (queue_arena == 0L) { return FALSE; } nar = RepAppl(Deref(ARG1)) + 1; nar[QUEUE_ARENA] = queue_arena; return true; } static Int p_nb_queue(USES_REGS1) { UInt arena_sz = (ASP - HR) / 16; if (LOCAL_DepthArenas > 1) arena_sz /= LOCAL_DepthArenas; if (arena_sz < MIN_ARENA_SIZE) arena_sz = MIN_ARENA_SIZE; if (arena_sz > MAX_ARENA_SIZE) arena_sz = MAX_ARENA_SIZE; return nb_queue(arena_sz PASS_REGS); } static Int p_nb_queue_sized(USES_REGS1) { Term t = Deref(ARG2); if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "nb_queue"); return FALSE; } if (!IsIntegerTerm(t)) { Yap_Error(TYPE_ERROR_INTEGER, t, "nb_queue"); return FALSE; } return nb_queue((UInt)IntegerOfTerm(t) PASS_REGS); } static CELL *GetQueue(Term t, char *caller) { t = Deref(t); if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, caller); return NULL; } if (!IsApplTerm(t)) { Yap_Error(TYPE_ERROR_COMPOUND, t, caller); return NULL; } if (FunctorOfTerm(t) != FunctorNBQueue) { Yap_Error(DOMAIN_ERROR_ARRAY_TYPE, t, caller); return NULL; } return RepAppl(t) + 1; } static Term GetQueueArena(CELL *qd, char *caller) { Term t = Deref(qd[QUEUE_ARENA]); if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, caller); return 0L; } if (!IsApplTerm(t)) { Yap_Error(TYPE_ERROR_COMPOUND, t, caller); return 0L; } if (FunctorOfTerm(t) != FunctorBigInt) { Yap_Error(DOMAIN_ERROR_ARRAY_TYPE, t, caller); return 0L; } return t; } static void RecoverArena(Term arena USES_REGS) { CELL *pt = ArenaPt(arena), *max = ArenaLimit(arena); if (max == HR) { HR = pt; } } static Int p_nb_queue_close(USES_REGS1) { Term t = Deref(ARG1); Int out; LOCAL_DepthArenas--; if (!IsVarTerm(t)) { CELL *qp; qp = GetQueue(t, "queue/3"); if (qp == NULL) { return Yap_unify(ARG3, ARG2); } if (qp[QUEUE_ARENA] != MkIntTerm(0)) RecoverArena(qp[QUEUE_ARENA] PASS_REGS); if (qp[QUEUE_SIZE] == MkIntTerm(0)) { return Yap_unify(ARG3, ARG2); } out = Yap_unify(ARG3, qp[QUEUE_TAIL]) && Yap_unify(ARG2, qp[QUEUE_HEAD]); RESET_VARIABLE(qp + QUEUE_TAIL); qp[QUEUE_HEAD] = qp[QUEUE_TAIL] = (CELL)(qp + QUEUE_TAIL); qp[QUEUE_SIZE] = MkIntTerm(0); return out; } Yap_Error(INSTANTIATION_ERROR, t, "queue/3"); return FALSE; } static Int p_nb_queue_enqueue(USES_REGS1) { CELL *qd = GetQueue(ARG1, "enqueue"); UInt old_sz; Term arena, qsize, to; UInt min_size; if (!qd) return FALSE; arena = GetQueueArena(qd, "enqueue"); if (arena == 0L) { return FALSE; } if (IsPairTerm(qd[QUEUE_HEAD])) { min_size = ArenaPt(arena) - RepPair(qd[QUEUE_HEAD]); } else { min_size = 0L; } to = CopyTermToArena(ARG2, arena, FALSE, TRUE, 2, qd + QUEUE_ARENA, min_size PASS_REGS); if (to == 0L) return FALSE; cell_space_t cspace; qd = GetQueue(ARG1, "enqueue"); arena = GetQueueArena(qd, "enqueue"); /* garbage collection ? */ enter_cell_space(&cspace); HR = HB = ArenaPt(arena); old_sz = ArenaSz(arena); qsize = IntegerOfTerm(qd[QUEUE_SIZE]); qd[QUEUE_SIZE] = Global_MkIntegerTerm(qsize + 1); if (qsize == 0) { qd[QUEUE_HEAD] = AbsPair(HR); } else { *VarOfTerm(qd[QUEUE_TAIL]) = AbsPair(HR); } *HR++ = to; RESET_VARIABLE(HR); qd[QUEUE_TAIL] = (CELL)HR; HR++; CloseArena(&cspace, qd + QUEUE_ARENA, old_sz PASS_REGS); return TRUE; } static Int p_nb_queue_dequeue(USES_REGS1) { CELL *qd = GetQueue(ARG1, "dequeue"); UInt old_sz, qsz; Term arena, out; cell_space_t cspace; if (!qd) return FALSE; qsz = IntegerOfTerm(qd[QUEUE_SIZE]); if (qsz == 0) return FALSE; arena = GetQueueArena(qd, "dequeue"); if (arena == 0L) return FALSE; old_sz = ArenaSz(arena); out = HeadOfTerm(qd[QUEUE_HEAD]); qd[QUEUE_HEAD] = TailOfTerm(qd[QUEUE_HEAD]); /* garbage collection ? */ enter_cell_space(&cspace); qd[QUEUE_SIZE] = Global_MkIntegerTerm(qsz - 1); CloseArena(&cspace, &arena, old_sz PASS_REGS); return Yap_unify(out, ARG2); } /* purge an entry from the queue, replacing it by [] */ static Int p_nb_queue_replace(USES_REGS1) { CELL *qd = GetQueue(ARG1, "dequeue"); UInt qsz; Term queue, t = Deref(ARG2); if (!qd) return FALSE; qsz = IntegerOfTerm(qd[QUEUE_SIZE]); if (qsz == 0) return FALSE; queue = qd[QUEUE_HEAD]; for (; qsz > 0; qsz--) { if (Yap_eq(HeadOfTerm(queue), t)) { *RepPair(Deref(queue)) = Deref(ARG3); return TRUE; } queue = TailOfTerm(queue); } return FALSE; } static Int p_nb_queue_peek(USES_REGS1) { CELL *qd = GetQueue(ARG1, "queue_peek"); UInt qsz; if (!qd) return FALSE; qsz = IntegerOfTerm(qd[QUEUE_SIZE]); if (qsz == 0) return FALSE; return Yap_unify(HeadOfTerm(qd[QUEUE_HEAD]), ARG2); } static Int p_nb_queue_empty(USES_REGS1) { CELL *qd = GetQueue(ARG1, "queue_empty"); if (!qd) return FALSE; return (IntegerOfTerm(qd[QUEUE_SIZE]) == 0); } static Int p_nb_queue_size(USES_REGS1) { CELL *qd = GetQueue(ARG1, "queue_size"); if (!qd) return FALSE; return Yap_unify(ARG2, qd[QUEUE_SIZE]); } static Int p_nb_queue_show(USES_REGS1) { CELL *qd = GetQueue(ARG1, "queue_size"); if (!qd) return FALSE; return Yap_unify(ARG2, qd[QUEUE_HEAD]); } static CELL *GetHeap(Term t, char *caller) { t = Deref(t); if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, caller); return NULL; } if (!IsApplTerm(t)) { Yap_Error(TYPE_ERROR_COMPOUND, t, caller); return NULL; } return RepAppl(t) + 1; } static Term MkZeroApplTerm(Functor f, UInt sz USES_REGS) { Term t0, tf; CELL *pt; if (HR + (sz + 1) > ASP - 1024) return TermNil; tf = AbsAppl(HR); *HR = (CELL)f; t0 = MkIntTerm(0); pt = HR + 1; while (sz--) { *pt++ = t0; } HR = pt; return tf; } static Int p_nb_heap(USES_REGS1) { Term heap_arena, heap, *ar, *nar; UInt hsize; Term tsize = Deref(ARG1); UInt arena_sz = (ASP-HR) / 16; if (IsVarTerm(tsize)) { Yap_Error(INSTANTIATION_ERROR, tsize, "nb_heap"); return FALSE; } else { if (!IsIntegerTerm(tsize)) { Yap_Error(TYPE_ERROR_INTEGER, tsize, "nb_heap"); return FALSE; } hsize = IntegerOfTerm(tsize); } if (arena_sz < hsize) arena_sz = hsize; while ((heap = MkZeroApplTerm( Yap_MkFunctor(AtomHeap, 2 * hsize + HEAP_START + 1), 2 * hsize + HEAP_START + 1 PASS_REGS)) == TermNil) { if (!Yap_gcl((2 * hsize + HEAP_START + 1) * sizeof(CELL), 2, ENV, P)) { Yap_Error(RESOURCE_ERROR_STACK, TermNil, LOCAL_ErrorMessage); return FALSE; } } if (!Yap_unify(heap, ARG2)) return FALSE; ar = RepAppl(heap) + 1; ar[HEAP_ARENA] = ar[HEAP_SIZE] = MkIntTerm(0); ar[HEAP_MAX] = tsize; if (arena_sz < 1024) arena_sz = 1024; cell_space_t cellSpace; enter_cell_space(&cellSpace); heap_arena = NewArena(arena_sz, worker_id, 1, NULL, &cellSpace); exit_cell_space(&cellSpace); if (heap_arena == 0L) { return FALSE; } nar = RepAppl(Deref(ARG2)) + 1; nar[HEAP_ARENA] = heap_arena; return TRUE; } static Int p_nb_heap_close(USES_REGS1) { Term t = Deref(ARG1); if (!IsVarTerm(t)) { CELL *qp; qp = RepAppl(t) + 1; if (qp[HEAP_ARENA] != MkIntTerm(0)) RecoverArena(qp[HEAP_ARENA] PASS_REGS); qp[-1] = (CELL)Yap_MkFunctor(AtomHeap, 1); qp[0] = MkIntegerTerm(0); return TRUE; } Yap_Error(INSTANTIATION_ERROR, t, "heap_close/1"); return FALSE; } static void PushHeap(CELL *pt, UInt off) { while (off) { UInt noff = (off + 1) / 2 - 1; if (Yap_compare_terms(pt[2 * off], pt[2 * noff]) < 0) { Term tk = pt[2 * noff]; Term tv = pt[2 * noff + 1]; pt[2 * noff] = pt[2 * off]; pt[2 * noff + 1] = pt[2 * off + 1]; pt[2 * off] = tk; pt[2 * off + 1] = tv; off = noff; } else { return; } } } static void DelHeapRoot(CELL *pt, UInt sz) { UInt indx = 0; Term tk, tv; sz--; tk = pt[2 * sz]; tv = pt[2 * sz + 1]; pt[2 * sz] = TermNil; pt[2 * sz + 1] = TermNil; while (TRUE) { if (sz < 2 * indx + 3 || Yap_compare_terms(pt[4 * indx + 2], pt[4 * indx + 4]) < 0) { if (sz < 2 * indx + 2 || Yap_compare_terms(tk, pt[4 * indx + 2]) < 0) { pt[2 * indx] = tk; pt[2 * indx + 1] = tv; return; } else { pt[2 * indx] = pt[4 * indx + 2]; pt[2 * indx + 1] = pt[4 * indx + 3]; indx = 2 * indx + 1; } } else { if (Yap_compare_terms(tk, pt[4 * indx + 4]) < 0) { pt[2 * indx] = tk; pt[2 * indx + 1] = tv; return; } else { pt[2 * indx] = pt[4 * indx + 4]; pt[2 * indx + 1] = pt[4 * indx + 5]; indx = 2 * indx + 2; } } } } static Int p_nb_heap_add_to_heap(USES_REGS1) { CELL *qd = GetHeap(ARG1, "add_to_heap"), *pt; UInt hsize, hmsize, old_sz; Term arena, to, key; UInt mingrow; cell_space_t cspace; if (!qd) return FALSE; restart: hsize = IntegerOfTerm(qd[HEAP_SIZE]); hmsize = IntegerOfTerm(qd[HEAP_MAX]); if (hsize == hmsize) { CELL *top = qd + (HEAP_START + 2 * hmsize); UInt extra_size; if (hmsize <= 64 * 1024) { extra_size = 64 * 1024; } else { extra_size = hmsize; } if ((extra_size = Yap_InsertInGlobal(top, extra_size * 2 * sizeof(CELL))) == 0) { Yap_Error(RESOURCE_ERROR_STACK, TermNil, "No Stack Space for Non-Backtrackable terms"); return FALSE; } enter_cell_space(&cspace); extra_size = extra_size / (2 * sizeof(CELL)); qd = GetHeap(ARG1, "add_to_heap"); hmsize += extra_size; if (!qd) return FALSE; qd[-1] = (CELL)Yap_MkFunctor(AtomHeap, 2 * hmsize + HEAP_START); top = qd + (HEAP_START + 2 * (hmsize - extra_size)); while (extra_size) { RESET_VARIABLE(top); RESET_VARIABLE(top + 1); top += 2; extra_size--; } arena = qd[HEAP_ARENA]; old_sz = ArenaSz(arena); HR = HB = ArenaPt(arena); qd[HEAP_MAX] = Global_MkIntegerTerm(hmsize); CloseArena(&cspace, qd + HEAP_ARENA, old_sz PASS_REGS); goto restart; } arena = qd[HEAP_ARENA]; if (arena == 0L) return FALSE; mingrow = garena_overflow_size(ArenaPt(arena) PASS_REGS); ARG2 = CopyTermToArena(ARG2, arena, FALSE, TRUE, 3, qd + HEAP_ARENA, mingrow PASS_REGS); qd = GetHeap(ARG1, "add_to_heap"); arena = qd[HEAP_ARENA]; to = CopyTermToArena(ARG3, arena, FALSE, TRUE, 3, qd + HEAP_ARENA, mingrow PASS_REGS); /* protect key in ARG2 in case there is an overflow while copying to */ key = ARG2; if (key == 0 || to == 0L) return FALSE; qd = GetHeap(ARG1, "add_to_heap"); arena = qd[HEAP_ARENA]; /* garbage collection ? */ enter_cell_space(&cspace); HR = HB = ArenaPt(arena); old_sz = ArenaSz(arena); while (old_sz < MIN_ARENA_SIZE) { UInt gsiz = hsize * 2; if (gsiz > 1024 * 1024) { gsiz = 1024 * 1024; } else if (gsiz < 1024) { gsiz = 1024; } ARG3 = to; if ((arena=GrowArena(arena, ArenaLimit(arena), old_sz, gsiz, 3, &cspace PASS_REGS))==0) { Yap_Error(RESOURCE_ERROR_STACK, arena, LOCAL_ErrorMessage); return 0L; } to = ARG3; qd = RepAppl(Deref(ARG1)) + 1; arena = qd[HEAP_ARENA]; HR = HB = ArenaPt(arena); old_sz = ArenaSz(arena); } pt = qd + HEAP_START; pt[2 * hsize] = key; pt[2 * hsize + 1] = to; PushHeap(pt, hsize); qd[HEAP_SIZE] = Global_MkIntegerTerm(hsize + 1); CloseArena(&cspace, qd + HEAP_ARENA, old_sz PASS_REGS); return TRUE; } static Int p_nb_heap_del(USES_REGS1) { CELL *qd = GetHeap(ARG1, "deheap"); UInt qsz; Term arena; Term tk, tv; if (!qd) return FALSE; qsz = IntegerOfTerm(qd[HEAP_SIZE]); if (qsz == 0) return FALSE; arena = qd[HEAP_ARENA]; if (arena == 0L) return FALSE; /* garbage collection ? */ qd[HEAP_SIZE] =MkIntTerm(qsz - 1); tk = qd[HEAP_START]; tv = qd[HEAP_START + 1]; DelHeapRoot(qd + HEAP_START, qsz); return Yap_unify(tk, ARG2) && Yap_unify(tv, ARG3); } static Int p_nb_heap_peek(USES_REGS1) { CELL *qd = GetHeap(ARG1, "heap_peek"); UInt qsz; Term tk, tv; if (!qd) return FALSE; qsz = IntegerOfTerm(qd[HEAP_SIZE]); if (qsz == 0) return FALSE; tk = qd[HEAP_START]; tv = qd[HEAP_START + 1]; return Yap_unify(tk, ARG2) && Yap_unify(tv, ARG3); } static Int p_nb_heap_empty(USES_REGS1) { CELL *qd = GetHeap(ARG1, "heap_empty"); if (!qd) return FALSE; return (IntegerOfTerm(qd[HEAP_SIZE]) == 0); } static Int p_nb_heap_size(USES_REGS1) { CELL *qd = GetHeap(ARG1, "heap_size"); if (!qd) return FALSE; return Yap_unify(ARG2, qd[HEAP_SIZE]); } static Int p_nb_beam(USES_REGS1) { cell_space_t cspace; Term beam_arena, beam, *ar, *nar; UInt hsize; Term tsize = Deref(ARG1); UInt arena_sz = (HR - H0) / 16; if (IsVarTerm(tsize)) { Yap_Error(INSTANTIATION_ERROR, tsize, "nb_beam"); return FALSE; } else { if (!IsIntegerTerm(tsize)) { Yap_Error(TYPE_ERROR_INTEGER, tsize, "nb_beam"); return FALSE; } hsize = IntegerOfTerm(tsize); } while ((beam = MkZeroApplTerm( Yap_MkFunctor(AtomHeap, 5 * hsize + HEAP_START + 1), 5 * hsize + HEAP_START + 1 PASS_REGS)) == TermNil) { if (!Yap_gcl((4 * hsize + HEAP_START + 1) * sizeof(CELL), 2, ENV, P)) { Yap_Error(RESOURCE_ERROR_STACK, TermNil, LOCAL_ErrorMessage); return FALSE; } } if (!Yap_unify(beam, ARG2)) return FALSE; ar = RepAppl(beam) + 1; ar[HEAP_ARENA] = ar[HEAP_SIZE] = MkIntTerm(0); ar[HEAP_MAX] = tsize; if (arena_sz < 1024) arena_sz = 1024; beam_arena = NewArena(arena_sz, worker_id, 1, NULL, &cspace); if (beam_arena == 0L) { return FALSE; } nar = RepAppl(Deref(ARG2)) + 1; nar[HEAP_ARENA] = beam_arena; return TRUE; } static Int p_nb_beam_close(USES_REGS1) { return p_nb_heap_close(PASS_REGS1); } /* we have two queues, one with Key, IndxQueue2 the other with Key, IndxQueue1, Val */ static void PushBeam(CELL *pt, CELL *npt, UInt hsize, Term key, Term to) { CACHE_REGS UInt off = hsize, off2 = hsize; Term toff, toff2; /* push into first queue */ while (off) { UInt noff = (off + 1) / 2 - 1; if (Yap_compare_terms(key, pt[2 * noff]) < 0) { UInt i2 = IntegerOfTerm(pt[2 * noff + 1]); pt[2 * off] = pt[2 * noff]; pt[2 * off + 1] = pt[2 * noff + 1]; npt[3 * i2 + 1] = Global_MkIntegerTerm(off); off = noff; } else { break; } } toff = Global_MkIntegerTerm(off); /* off says where we are in first queue */ /* push into second queue */ while (off2) { UInt noff = (off2 + 1) / 2 - 1; if (Yap_compare_terms(key, npt[3 * noff]) > 0) { UInt i1 = IntegerOfTerm(npt[3 * noff + 1]); npt[3 * off2] = npt[3 * noff]; npt[3 * off2 + 1] = npt[3 * noff + 1]; npt[3 * off2 + 2] = npt[3 * noff + 2]; pt[2 * i1 + 1] = Global_MkIntegerTerm(off2); off2 = noff; } else { break; } } toff2 = Global_MkIntegerTerm(off2); /* store elements in their rightful place */ npt[3 * off2] = pt[2 * off] = key; pt[2 * off + 1] = toff2; npt[3 * off2 + 1] = toff; npt[3 * off2 + 2] = to; } static void DelBeamMax(CELL *pt, CELL *pt2, UInt sz) { CACHE_REGS UInt off = IntegerOfTerm(pt2[1]); UInt indx = 0; Term tk, ti, tv; sz--; /* first, fix the reverse queue */ tk = pt2[3 * sz]; ti = pt2[3 * sz + 1]; tv = pt2[3 * sz + 2]; while (TRUE) { if (sz < 2 * indx + 3 || Yap_compare_terms(pt2[6 * indx + 3], pt2[6 * indx + 6]) > 0) { if (sz < 2 * indx + 2 || Yap_compare_terms(tk, pt2[6 * indx + 3]) > 0) { break; } else { UInt off = IntegerOfTerm(pt2[6 * indx + 4]); pt2[3 * indx] = pt2[6 * indx + 3]; pt2[3 * indx + 1] = pt2[6 * indx + 4]; pt2[3 * indx + 2] = pt2[6 * indx + 5]; pt[2 * off + 1] = Global_MkIntegerTerm(indx); indx = 2 * indx + 1; } } else { if (Yap_compare_terms(tk, pt2[6 * indx + 6]) > 0) { break; } else { UInt off = IntegerOfTerm(pt2[6 * indx + 7]); pt2[3 * indx] = pt2[6 * indx + 6]; pt2[3 * indx + 1] = pt2[6 * indx + 7]; pt2[3 * indx + 2] = pt2[6 * indx + 8]; pt[2 * off + 1] = Global_MkIntegerTerm(indx); indx = 2 * indx + 2; } } } pt[2 * IntegerOfTerm(ti) + 1] = Global_MkIntegerTerm(indx); pt2[3 * indx] = tk; pt2[3 * indx + 1] = ti; pt2[3 * indx + 2] = tv; /* now, fix the standard queue */ if (off != sz) { Term toff, toff2, key; UInt off2; key = pt[2 * sz]; toff2 = pt[2 * sz + 1]; off2 = IntegerOfTerm(toff2); /* off says where we are in first queue */ /* push into second queue */ while (off) { UInt noff = (off + 1) / 2 - 1; if (Yap_compare_terms(key, pt[2 * noff]) < 0) { UInt i1 = IntegerOfTerm(pt[2 * noff + 1]); pt[2 * off] = pt[2 * noff]; pt[2 * off + 1] = pt[2 * noff + 1]; pt2[3 * i1 + 1] = Global_MkIntegerTerm(off); off = noff; } else { break; } } toff = Global_MkIntegerTerm(off); /* store elements in their rightful place */ pt[2 * off] = key; pt2[3 * off2 + 1] = toff; pt[2 * off + 1] = toff2; } } static Term DelBeamMin(CELL *pt, CELL *pt2, UInt sz) { CACHE_REGS UInt off2 = IntegerOfTerm(pt[1]); Term ov = pt2[3 * off2 + 2]; /* return value */ UInt indx = 0; Term tk, tv; sz--; /* first, fix the standard queue */ tk = pt[2 * sz]; tv = pt[2 * sz + 1]; while (TRUE) { if (sz < 2 * indx + 3 || Yap_compare_terms(pt[4 * indx + 2], pt[4 * indx + 4]) < 0) { if (sz < 2 * indx + 2 || Yap_compare_terms(tk, pt[4 * indx + 2]) < 0) { break; } else { UInt off2 = IntegerOfTerm(pt[4 * indx + 3]); pt[2 * indx] = pt[4 * indx + 2]; pt[2 * indx + 1] = pt[4 * indx + 3]; pt2[3 * off2 + 1] = Global_MkIntegerTerm(indx); indx = 2 * indx + 1; } } else { if (Yap_compare_terms(tk, pt[4 * indx + 4]) < 0) { break; } else { UInt off2 = IntegerOfTerm(pt[4 * indx + 5]); pt[2 * indx] = pt[4 * indx + 4]; pt[2 * indx + 1] = pt[4 * indx + 5]; pt2[3 * off2 + 1] = Global_MkIntegerTerm(indx); indx = 2 * indx + 2; } } } pt[2 * indx] = tk; pt[2 * indx + 1] = tv; pt2[3 * IntegerOfTerm(tv) + 1] = Global_MkIntegerTerm(indx); /* now, fix the reverse queue */ if (off2 != sz) { Term to, toff, toff2, key; UInt off; key = pt2[3 * sz]; toff = pt2[3 * sz + 1]; to = pt2[3 * sz + 2]; off = IntegerOfTerm(toff); /* off says where we are in first queue */ /* push into second queue */ while (off2) { UInt noff = (off2 + 1) / 2 - 1; if (Yap_compare_terms(key, pt2[3 * noff]) > 0) { UInt i1 = IntegerOfTerm(pt2[3 * noff + 1]); pt2[3 * off2] = pt2[3 * noff]; pt2[3 * off2 + 1] = pt2[3 * noff + 1]; pt2[3 * off2 + 2] = pt2[3 * noff + 2]; pt[2 * i1 + 1] = Global_MkIntegerTerm(off2); off2 = noff; } else { break; } } toff2 = Global_MkIntegerTerm(off2); /* store elements in their rightful place */ pt2[3 * off2] = key; pt[2 * off + 1] = toff2; pt2[3 * off2 + 1] = toff; pt2[3 * off2 + 2] = to; } return ov; } static Int p_nb_beam_add_to_beam(USES_REGS1) { CELL *qd = GetHeap(ARG1, "add_to_beam"), *pt; UInt hsize, hmsize, old_sz; Term arena, to, qsize, key; UInt mingrow; cell_space_t cspace; if (!qd) return FALSE; hsize = IntegerOfTerm(qd[HEAP_SIZE]); hmsize = IntegerOfTerm(qd[HEAP_MAX]); key = Deref(ARG2); if (hsize == hmsize) { pt = qd + HEAP_START; if (Yap_compare_terms(pt[2 * hmsize], Deref(ARG2)) > 0) { /* smaller than current max, we need to drop current max */ DelBeamMax(pt, pt + 2 * hmsize, hmsize); hsize--; } else { return TRUE; } } arena = qd[HEAP_ARENA]; if (arena == 0L) return FALSE; mingrow = garena_overflow_size(ArenaPt(arena) PASS_REGS); key = CopyTermToArena(ARG2, qd[HEAP_ARENA], FALSE, TRUE, 3, qd + HEAP_ARENA, mingrow PASS_REGS); arena = qd[HEAP_ARENA]; to = CopyTermToArena(ARG3, arena, FALSE, TRUE, 3, qd + HEAP_ARENA, mingrow PASS_REGS); if (key == 0 || to == 0L) return FALSE; qd = GetHeap(ARG1, "add_to_beam"); arena = qd[HEAP_ARENA]; /* garbage collection ? */ enter_cell_space(&cspace); HR = HB = ArenaPt(arena); old_sz = ArenaSz(arena); qsize = IntegerOfTerm(qd[QUEUE_SIZE]); while (old_sz < MIN_ARENA_SIZE) { UInt gsiz = HR - RepPair(qd[QUEUE_HEAD]); if (gsiz > 1024 * 1024) { gsiz = 1024 * 1024; } else if (gsiz < 1024) { gsiz = 1024; } ARG3 = to; /* fprintf(stderr,"growing %ld cells\n",(unsigned long int)gsiz);*/ if (!GrowArena(arena, ArenaLimit(arena), old_sz, gsiz, 3, &cspace PASS_REGS)) { Yap_Error(RESOURCE_ERROR_STACK, arena, LOCAL_ErrorMessage); return 0L; } to = ARG3; qd = RepAppl(Deref(ARG1)) + 1; arena = qd[HEAP_ARENA]; old_sz = ArenaSz(arena); } pt = qd + HEAP_START; PushBeam(pt, pt + 2 * hmsize, hsize, key, to); qd[HEAP_SIZE] = Global_MkIntegerTerm(hsize + 1); CloseArena(&cspace, qd + HEAP_ARENA, old_sz PASS_REGS); return TRUE; } static Int p_nb_beam_del(USES_REGS1) { CELL *qd = GetHeap(ARG1, "debeam"); UInt old_sz, qsz; Term arena; Term tk, tv; cell_space_t cspace; if (!qd) return FALSE; qsz = IntegerOfTerm(qd[HEAP_SIZE]); if (qsz == 0) return FALSE; arena = qd[HEAP_ARENA]; if (arena == 0L) return FALSE; old_sz = ArenaSz(arena); /* garbage collection ? */ enter_cell_space(&cspace); qd[HEAP_SIZE] = Global_MkIntegerTerm(qsz - 1); CloseArena(&cspace, &arena, old_sz PASS_REGS); tk = qd[HEAP_START]; tv = DelBeamMin(qd + HEAP_START, qd + (HEAP_START + 2 * IntegerOfTerm(qd[HEAP_MAX])), qsz); return Yap_unify(tk, ARG2) && Yap_unify(tv, ARG3); } #ifdef DEBUG static Int p_nb_beam_check(USES_REGS1) { CELL *qd = GetHeap(ARG1, "debeam"); UInt qsz, qmax; CELL *pt, *pt2; UInt i; if (!qd) return FALSE; qsz = IntegerOfTerm(qd[HEAP_SIZE]); qmax = IntegerOfTerm(qd[HEAP_MAX]); if (qsz == 0) return TRUE; pt = qd + HEAP_START; pt2 = pt + 2 * qmax; for (i = 1; i < qsz; i++) { UInt back; if (Yap_compare_terms(pt[2 * ((i + 1) / 2 - 1)], pt[2 * i]) > 0) { Yap_DebugPlWrite(pt[2 * ((i + 1) / 2 - 1)]); fprintf(stderr, "\n"); Yap_DebugPlWrite(pt[2 * i]); fprintf(stderr, "\n"); fprintf(stderr, "Error at %ld\n", (unsigned long int)i); return FALSE; } back = IntegerOfTerm(pt[2 * i + 1]); if (IntegerOfTerm(pt2[3 * back + 1]) != i) { fprintf(stderr, "Link error at %ld\n", (unsigned long int)i); return FALSE; } } for (i = 1; i < qsz; i++) { if (Yap_compare_terms(pt2[3 * ((i + 1) / 2 - 1)], pt2[3 * i]) < 0) { fprintf(stderr, "Error at sec %ld\n", (unsigned long int)i); Yap_DebugPlWrite(pt2[3 * ((i + 1) / 2 - 1)]); fprintf(stderr, "\n"); Yap_DebugPlWrite(pt2[3 * i]); fprintf(stderr, "\n"); return FALSE; } } return TRUE; } #endif static Int p_nb_beam_keys(USES_REGS1) { CELL *qd; UInt qsz; CELL *pt, *ho; UInt i; restart: qd = GetHeap(ARG1, "beam_keys"); if (!qd) return FALSE; qsz = IntegerOfTerm(qd[HEAP_SIZE]); ho = HR; pt = qd + HEAP_START; if (qsz == 0) return Yap_unify(ARG2, TermNil); for (i = 0; i < qsz; i++) { if (HR > ASP - 1024) { HR = ho; if (!Yap_gcl(((ASP - HR) - 1024) * sizeof(CELL), 2, ENV, P)) { Yap_Error(RESOURCE_ERROR_STACK, TermNil, LOCAL_ErrorMessage); return TermNil; } goto restart; } *HR++ = pt[0]; *HR = AbsPair(HR + 1); HR++; pt += 2; } HR[-1] = TermNil; return Yap_unify(ARG2, AbsPair(ho)); } static Int p_nb_beam_peek(USES_REGS1) { CELL *qd = GetHeap(ARG1, "beam_peek"), *pt, *pt2; UInt qsz, qbsize; Term tk, tv; if (!qd) return FALSE; qsz = IntegerOfTerm(qd[HEAP_SIZE]); qbsize = IntegerOfTerm(qd[HEAP_MAX]); if (qsz == 0) return FALSE; pt = qd + HEAP_START; pt2 = pt + 2 * qbsize; tk = pt[0]; tv = pt2[2]; return Yap_unify(tk, ARG2) && Yap_unify(tv, ARG3); } static Int p_nb_beam_empty(USES_REGS1) { CELL *qd = GetHeap(ARG1, "beam_empty"); if (!qd) return FALSE; return (IntegerOfTerm(qd[HEAP_SIZE]) == 0); } static Int p_nb_beam_size(USES_REGS1) { CELL *qd = GetHeap(ARG1, "beam_size"); if (!qd) return FALSE; return Yap_unify(ARG2, qd[HEAP_SIZE]); } /// @} static Int cont_current_nb(USES_REGS1) { Int unif; GlobalEntry *ge = (GlobalEntry *)IntegerOfTerm(EXTRA_CBACK_ARG(1, 1)); unif = Yap_unify(MkAtomTerm(AbsAtom(ge->AtomOfGE)), ARG1); ge = ge->NextGE; if (!ge) { if (unif) cut_succeed(); else cut_fail(); } else { EXTRA_CBACK_ARG(1, 1) = MkIntegerTerm((Int)ge); return unif; } } static Int init_current_nb(USES_REGS1) { /* current_atom(?Atom) */ Term t1 = Deref(ARG1); if (!IsVarTerm(t1)) { if (IsAtomTerm(t1)) { if (!FindGlobalEntry(AtomOfTerm(t1) PASS_REGS)) { cut_fail(); } else { cut_succeed(); } } else { Yap_Error(TYPE_ERROR_ATOM, t1, "nb_current"); cut_fail(); } } READ_LOCK(HashChain[0].AERWLock); EXTRA_CBACK_ARG(1, 1) = MkIntegerTerm((Int)LOCAL_GlobalVariables); return cont_current_nb(PASS_REGS1); } void Yap_InitGlobals(void) { CACHE_REGS Term cm = CurrentModule; Yap_InitCPred("$allocate_arena", 2, p_allocate_arena, 0); Yap_InitCPred("arena_size", 1, p_default_arena_size, 0); Yap_InitCPred("b_setval", 2, p_b_setval, SafePredFlag); Yap_InitCPred("__B_setval__", 2, p_b_setval, HiddenPredFlag | SafePredFlag); /** @pred b_setval(+ _Name_, + _Value_) Associate the term _Value_ with the atom _Name_ or replaces the currently associated value with _Value_. If _Name_ does not refer to an existing global variable a variable with initial value [] is created (the empty list). On backtracking the assignment is reversed. */ /** @pred b_setval(+ _Name_,+ _Value_) Associate the term _Value_ with the atom _Name_ or replaces the currently associated value with _Value_. If _Name_ does not refer to an existing global variable a variable with initial value `[]` is created (the empty list). On backtracking the assignment is reversed. */ Yap_InitCPred("nb_setval", 2, p_nb_setval, 0L); Yap_InitCPred("__NB_setval__", 2, p_nb_setval, HiddenPredFlag); /** @pred nb_setval(+ _Name_, + _Value_) Associates a copy of _Value_ created with duplicate_term/2 with the atom _Name_. Note that this can be used to set an initial value other than `[]` prior to backtrackable assignment. */ /** @pred nb_setval(+ _Name_,+ _Value_) Associates a copy of _Value_ created with duplicate_term/2 with the atom _Name_. Note that this can be used to set an initial value other than `[]` prior to backtrackable assignment. */ Yap_InitCPred("nb_set_shared_val", 2, p_nb_set_shared_val, 0L); /** @pred nb_set_shared_val(+ _Name_, + _Value_) Associates the term _Value_ with the atom _Name_, but sharing non-backtrackable terms. This may be useful if you want to rewrite a global variable so that the new copy will survive backtracking, but you want to share structure with the previous term. The next example shows the differences between the three built-ins: ~~~~~ ?- nb_setval(a,a(_)),nb_getval(a,A),nb_setval(b,t(C,A)),nb_getval(b,B). A = a(_A), B = t(_B,a(_C)) ? ?- nb_setval(a,a(_)),nb_getval(a,A),nb_set_shared_val(b,t(C,A)),nb_getval(b,B). ?- nb_setval(a,a(_)),nb_getval(a,A),nb_linkval(b,t(C,A)),nb_getval(b,B). A = a(_A), B = t(C,a(_A)) ? ~~~~~ */ Yap_InitCPred("nb_linkval", 2, p_nb_linkval, 0L); /** @pred nb_linkval(+ _Name_, + _Value_) Associates the term _Value_ with the atom _Name_ without copying it. This is a fast special-purpose variation of nb_setval/2 intended for expert users only because the semantics on backtracking to a point before creating the link are poorly defined for compound terms. The principal term is always left untouched, but backtracking behaviour on arguments is undone if the original assignment was trailed and left alone otherwise, which implies that the history that created the term affects the behaviour on backtracking. Please consider the following example: ~~~~~ demo_nb_linkval :- T = nice(N), ( N = world, nb_linkval(myvar, T), fail ; nb_getval(myvar, V), writeln(V) ). ~~~~~ */ Yap_InitCPred("$nb_getval", 3, p_nb_getval, SafePredFlag); Yap_InitCPred("__NB_getval__", 3, p_nb_getval, HiddenPredFlag); Yap_InitCPred("__B_getval__", 3, p_nb_getval, HiddenPredFlag); Yap_InitCPred("nb_setarg", 3, p_nb_setarg, 0L); /** @pred nb_setarg(+{Arg], + _Term_, + _Value_) Assigns the _Arg_-th argument of the compound term _Term_ with the given _Value_ as setarg/3, but on backtracking the assignment is not reversed. If _Term_ is not atomic, it is duplicated using duplicate_term/2. This predicate uses the same technique as nb_setval/2. We therefore refer to the description of nb_setval/2 for details on non-backtrackable assignment of terms. This predicate is compatible to GNU-Prolog `setarg(A,T,V,false)`, removing the type-restriction on _Value_. See also nb_linkarg/3. Below is an example for counting the number of solutions of a goal. Note that this implementation is thread-safe, reentrant and capable of handling exceptions. Realising these features with a traditional implementation based on assert/retract or flag/3 is much more complicated. ~~~~~ succeeds_n_times(Goal, Times) :- Counter = counter(0), ( Goal, arg(1, Counter, N0), N is N0 + 1, nb_setarg(1, Counter, N), fail ; arg(1, Counter, Times) ). ~~~~~ */ Yap_InitCPred("nb_set_shared_arg", 3, p_nb_set_shared_arg, 0L); /** @pred nb_set_shared_arg(+ _Arg_, + _Term_, + _Value_) As nb_setarg/3, but like nb_linkval/2 it does not duplicate the global sub-terms in _Value_. Use with extreme care and consult the documentation of nb_linkval/2 before use. */ Yap_InitCPred("nb_linkarg", 3, p_nb_linkarg, 0L); /** @pred nb_linkarg(+ _Arg_, + _Term_, + _Value_) As nb_setarg/3, but like nb_linkval/2 it does not duplicate _Value_. Use with extreme care and consult the documentation of nb_linkval/2 before use. */ Yap_InitCPred("nb_delete", 1, p_nb_delete, 0L); /** @pred nb_delete(+ _Name_) Delete the named global variable. Global variables have been introduced by various Prolog implementations recently. We follow the implementation of them in SWI-Prolog, itself based on hProlog by Bart Demoen. GNU-Prolog provides a rich set of global variables, including arrays. Arrays can be implemented easily in YAP and SWI-Prolog using functor/3 and `setarg/3` due to the unrestricted arity of compound terms. @} */ Yap_InitCPred("nb_create", 3, p_nb_create, 0L); Yap_InitCPred("nb_create", 4, p_nb_create2, 0L); Yap_InitCPredBack("$nb_current", 1, 1, init_current_nb, cont_current_nb, SafePredFlag); /// @{ /// @addtogroup nb CurrentModule = GLOBALS_MODULE; Yap_InitCPred("nb_queue", 1, p_nb_queue, 0L); Yap_InitCPred("nb_queue", 2, p_nb_queue_sized, 0L); Yap_InitCPred("nb_queue_close", 3, p_nb_queue_close, SafePredFlag); Yap_InitCPred("nb_queue_enqueue", 2, p_nb_queue_enqueue, 0L); Yap_InitCPred("nb_queue_dequeue", 2, p_nb_queue_dequeue, SafePredFlag); Yap_InitCPred("nb_queue_peek", 2, p_nb_queue_peek, SafePredFlag); Yap_InitCPred("nb_queue_empty", 1, p_nb_queue_empty, SafePredFlag); Yap_InitCPred("nb_queue_replace", 3, p_nb_queue_replace, SafePredFlag); Yap_InitCPred("nb_queue_size", 2, p_nb_queue_size, SafePredFlag); Yap_InitCPred("nb_queue_show", 2, p_nb_queue_show, SafePredFlag); Yap_InitCPred("nb_heap", 2, p_nb_heap, 0L); Yap_InitCPred("nb_heap_close", 1, p_nb_heap_close, SafePredFlag); Yap_InitCPred("nb_heap_add", 3, p_nb_heap_add_to_heap, 0L); Yap_InitCPred("nb_heap_del", 3, p_nb_heap_del, SafePredFlag); Yap_InitCPred("nb_heap_peek", 3, p_nb_heap_peek, SafePredFlag); Yap_InitCPred("nb_heap_empty", 1, p_nb_heap_empty, SafePredFlag); Yap_InitCPred("nb_heap_size", 2, p_nb_heap_size, SafePredFlag); Yap_InitCPred("nb_beam", 2, p_nb_beam, 0L); Yap_InitCPred("nb_beam_close", 1, p_nb_beam_close, SafePredFlag); Yap_InitCPred("nb_beam_add", 3, p_nb_beam_add_to_beam, 0L); Yap_InitCPred("nb_beam_del", 3, p_nb_beam_del, SafePredFlag); Yap_InitCPred("nb_beam_peek", 3, p_nb_beam_peek, SafePredFlag); Yap_InitCPred("nb_beam_empty", 1, p_nb_beam_empty, SafePredFlag); Yap_InitCPred("nb_beam_keys", 2, p_nb_beam_keys, 0L); Yap_InitCPred("nb_create_accumulator", 2, p_nb_create_accumulator, 0L); Yap_InitCPred("nb_add_to_accumulator", 2, p_nb_add_to_accumulator, 0L); Yap_InitCPred("nb_accumulator_value", 2, p_nb_accumulator_value, 0L); #ifdef DEBUG Yap_InitCPred("nb_beam_check", 1, p_nb_beam_check, SafePredFlag); #endif Yap_InitCPred("nb_beam_size", 2, p_nb_beam_size, SafePredFlag); /// @} CurrentModule = cm; } /** @} */