| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | /*************************************************************************
 | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *	 YAP Prolog 							 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *	Yap Prolog was developed at NCCUP - Universidade do Porto	 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | ************************************************************************** | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * File:		unify.c							 * | 
					
						
							|  |  |  | * Last rev:								 * | 
					
						
							|  |  |  | * mods:									 * | 
					
						
							|  |  |  | * comments:	Unification and other auxiliary routines for absmi       * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *************************************************************************/ | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | /** @defgroup Rational_Trees Rational Trees
 | 
					
						
							| 
									
										
										
										
											2015-01-04 23:58:23 +00:00
										 |  |  | @ingroup extensions | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | @{ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Prolog unification is not a complete implementation. For efficiency | 
					
						
							|  |  |  | considerations, Prolog systems do not perform occur checks while | 
					
						
							|  |  |  | unifying terms. As an example, `X = a(X)` will not fail but instead | 
					
						
							|  |  |  | will create an infinite term of the form `a(a(a(a(a(...)))))`, or | 
					
						
							|  |  |  | <em>rational tree</em>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Rational trees are now supported by default in YAP. In previous | 
					
						
							|  |  |  | versions, this was not the default and these terms could easily lead | 
					
						
							|  |  |  | to infinite computation. For example, `X = a(X), X = X` would | 
					
						
							|  |  |  | enter an infinite loop. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The `RATIONAL_TREES` flag improves support for these | 
					
						
							|  |  |  | terms. Internal primitives are now aware that these terms can exist, and | 
					
						
							|  |  |  | will not enter infinite loops. Hence, the previous unification will | 
					
						
							|  |  |  | succeed. Another example, `X = a(X), ground(X)` will succeed | 
					
						
							|  |  |  | instead of looping. Other affected built-ins include the term comparison | 
					
						
							|  |  |  | primitives, numbervars/3, copy_term/2, and the internal | 
					
						
							|  |  |  | data base routines. The support does not extend to Input/Output routines | 
					
						
							|  |  |  | or to assert/1 YAP does not allow directly reading | 
					
						
							|  |  |  | rational trees, and you need to use `write_depth/2` to avoid | 
					
						
							|  |  |  | entering an infinite cycle when trying to write an infinite term. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-11 17:38:10 +00:00
										 |  |  | #define IN_UNIFY_C 1
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | #define HAS_CACHE_REGS 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #include "absmi.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 17:15:04 -05:00
										 |  |  | int   Yap_rational_tree_loop(CELL *, CELL *, CELL **, CELL **); | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 17:15:04 -05:00
										 |  |  | static int    OCUnify_complex(CELL *, CELL *, CELL *); | 
					
						
							|  |  |  | static int    OCUnify(register CELL, register CELL); | 
					
						
							|  |  |  | static Int   p_ocunify( USES_REGS1 ); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | /* support for rational trees and unification with occur checking */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | #define to_visit_base ((struct v_record *)AuxSp)
 | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-26 16:02:25 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | Yap_rational_tree_loop(CELL *pt0, CELL *pt0_end, CELL **to_visit, CELL **to_visit_max) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-05 10:27:51 +00:00
										 |  |  |   CELL ** base = to_visit; | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | rtree_loop: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   while (pt0 < pt0_end) { | 
					
						
							| 
									
										
										
										
											2001-09-24 18:07:16 +00:00
										 |  |  |     register CELL *ptd0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     register CELL d0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-09-24 18:07:16 +00:00
										 |  |  |     ptd0 = ++pt0;  | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     pt0 = ptd0; | 
					
						
							|  |  |  |     d0 = *ptd0; | 
					
						
							|  |  |  |     deref_head(d0, rtree_loop_unk); | 
					
						
							|  |  |  |   rtree_loop_nvar: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (d0 == TermFoundVar) | 
					
						
							|  |  |  | 	goto cufail; | 
					
						
							|  |  |  |       if (IsPairTerm(d0)) { | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | 	to_visit -= 3; | 
					
						
							|  |  |  | 	if (to_visit < to_visit_max) { | 
					
						
							| 
									
										
										
										
											2012-03-15 22:37:13 +00:00
										 |  |  | 	  to_visit = Yap_shift_visit(to_visit, &to_visit_max, &base); | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	to_visit[0] = pt0; | 
					
						
							|  |  |  | 	to_visit[1] = pt0_end; | 
					
						
							| 
									
										
										
										
											2011-02-02 20:45:42 +00:00
										 |  |  | 	to_visit[2] = (CELL *)*pt0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	*pt0 = TermFoundVar; | 
					
						
							|  |  |  | 	pt0_end = (pt0 = RepPair(d0) - 1) + 2; | 
					
						
							|  |  |  | 	continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (IsApplTerm(d0)) { | 
					
						
							|  |  |  | 	register Functor f; | 
					
						
							|  |  |  | 	register CELL *ap2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* store the terms to visit */ | 
					
						
							|  |  |  | 	ap2 = RepAppl(d0); | 
					
						
							|  |  |  | 	f = (Functor) (*ap2); | 
					
						
							|  |  |  | 	/* compare functors */ | 
					
						
							|  |  |  | 	if (IsExtensionFunctor(f)) { | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | 	to_visit -= 3; | 
					
						
							|  |  |  | 	if (to_visit < to_visit_max) { | 
					
						
							| 
									
										
										
										
											2012-03-15 22:37:13 +00:00
										 |  |  | 	  to_visit = Yap_shift_visit(to_visit, &to_visit_max, &base); | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	to_visit[0] = pt0; | 
					
						
							|  |  |  | 	to_visit[1] = pt0_end; | 
					
						
							| 
									
										
										
										
											2011-02-02 20:45:42 +00:00
										 |  |  | 	to_visit[2] = (CELL *)*pt0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	*pt0 = TermFoundVar; | 
					
						
							|  |  |  | 	d0 = ArityOfFunctor(f); | 
					
						
							|  |  |  | 	pt0 = ap2; | 
					
						
							|  |  |  | 	pt0_end = ap2 + d0; | 
					
						
							|  |  |  | 	continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     derefa_body(d0, ptd0, rtree_loop_unk, rtree_loop_nvar); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* Do we still have compound terms to visit */ | 
					
						
							| 
									
										
										
										
											2011-02-05 10:27:51 +00:00
										 |  |  |   if (to_visit < base) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     pt0 = to_visit[0]; | 
					
						
							|  |  |  |     pt0_end = to_visit[1]; | 
					
						
							|  |  |  |     *pt0 = (CELL)to_visit[2]; | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  |     to_visit += 3; | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |     goto rtree_loop; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2005-08-17 13:35:52 +00:00
										 |  |  |   return FALSE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | cufail: | 
					
						
							|  |  |  |   /* we found an infinite term */ | 
					
						
							| 
									
										
										
										
											2011-02-05 10:27:51 +00:00
										 |  |  |   while (to_visit < (CELL **)base) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     CELL *pt0; | 
					
						
							|  |  |  |     pt0 = to_visit[0]; | 
					
						
							|  |  |  |     *pt0 = (CELL)to_visit[2]; | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  |     to_visit += 3; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |   return TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int | 
					
						
							|  |  |  | rational_tree(Term d0) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | CACHE_REGS | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |   CELL  **to_visit_max = (CELL **)AuxBase, **to_visit  = (CELL **)AuxSp; | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (IsPairTerm(d0)) { | 
					
						
							|  |  |  |     CELL *pt0 = RepPair(d0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |     return Yap_rational_tree_loop(pt0-1, pt0+1, to_visit, to_visit_max); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } else if (IsApplTerm(d0)) { | 
					
						
							|  |  |  |     CELL *pt0 = RepAppl(d0); | 
					
						
							|  |  |  |     Functor f = (Functor)(*pt0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-01-24 09:36:14 +00:00
										 |  |  |     if (IsExtensionFunctor(f)) | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |     return Yap_rational_tree_loop(pt0, pt0+ArityOfFunctor(f), to_visit, to_visit_max); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } else | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | static int  | 
					
						
							|  |  |  | OCUnify_complex(CELL *pt0, CELL *pt0_end, CELL *pt1) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | CACHE_REGS | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | #ifdef THREADS
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  |   register REGSTORE *regp = Yap_regp; | 
					
						
							|  |  |  | #define Yap_REGS (*regp)
 | 
					
						
							|  |  |  | #elif defined(SHADOW_REGS)
 | 
					
						
							|  |  |  | #if defined(B) || defined(TR)
 | 
					
						
							|  |  |  |   register REGSTORE *regp = &Yap_REGS; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | #define Yap_REGS (*regp)
 | 
					
						
							|  |  |  | #endif /* defined(B) || defined(TR) || defined(HB) */
 | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | #ifdef SHADOW_HB
 | 
					
						
							|  |  |  |   register CELL *HBREG = HB; | 
					
						
							|  |  |  | #endif /* SHADOW_HB */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   struct unif_record  *unif = (struct unif_record *)AuxBase; | 
					
						
							|  |  |  |   struct v_record *to_visit  = (struct v_record *)AuxSp; | 
					
						
							|  |  |  | #define unif_base ((struct unif_record *)AuxBase)
 | 
					
						
							| 
									
										
										
										
											2005-08-23 15:39:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | loop: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   while (pt0 < pt0_end) { | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |     register CELL *ptd0 = pt0+1;  | 
					
						
							|  |  |  |     register CELL d0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ++pt1; | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |     pt0 = ptd0; | 
					
						
							|  |  |  |     d0 = *ptd0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     deref_head(d0, unify_comp_unk); | 
					
						
							|  |  |  |   unify_comp_nvar: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       register CELL *ptd1 = pt1; | 
					
						
							|  |  |  |       register CELL d1 = *ptd1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       deref_head(d1, unify_comp_nvar_unk); | 
					
						
							|  |  |  |     unify_comp_nvar_nvar: | 
					
						
							|  |  |  |       if (d0 == d1) { | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 	if (Yap_rational_tree_loop(pt0-1, pt0, (CELL **)to_visit, (CELL **)unif)) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  goto cufail; | 
					
						
							|  |  |  | 	continue; | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |       } | 
					
						
							|  |  |  |       if (IsPairTerm(d0)) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	if (!IsPairTerm(d1)) { | 
					
						
							|  |  |  | 	  goto cufail; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* now link the two structures so that no one else will */ | 
					
						
							|  |  |  | 	/* come here */ | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 	/* store the terms to visit */ | 
					
						
							|  |  |  | 	if (RATIONAL_TREES || pt0 < pt0_end) { | 
					
						
							|  |  |  | 	  to_visit --; | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  | 	  unif++; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	  if ((void *)to_visit < (void *)unif) { | 
					
						
							| 
									
										
										
										
											2009-05-22 21:53:24 -05:00
										 |  |  | 	    CELL **urec = (CELL **)unif; | 
					
						
							| 
									
										
										
										
											2012-03-15 22:37:13 +00:00
										 |  |  | 	    to_visit = (struct v_record *)Yap_shift_visit((CELL **)to_visit, &urec, NULL); | 
					
						
							| 
									
										
										
										
											2009-05-22 21:53:24 -05:00
										 |  |  | 	    unif = (struct unif_record *)urec; | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	  to_visit->start0 = pt0; | 
					
						
							|  |  |  | 	  to_visit->end0 = pt0_end; | 
					
						
							|  |  |  | 	  to_visit->start1 = pt1; | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  | 	  unif[-1].old = *pt0; | 
					
						
							|  |  |  | 	  unif[-1].ptr = pt0; | 
					
						
							|  |  |  | 	  *pt0 = d1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	pt0_end = (pt0 = RepPair(d0) - 1) + 2; | 
					
						
							|  |  |  | 	pt1 = RepPair(d1) - 1; | 
					
						
							|  |  |  | 	continue; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |       if (IsApplTerm(d0)) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	register Functor f; | 
					
						
							|  |  |  | 	register CELL *ap2, *ap3; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!IsApplTerm(d1)) { | 
					
						
							|  |  |  | 	  goto cufail; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 	/* store the terms to visit */ | 
					
						
							|  |  |  | 	ap2 = RepAppl(d0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	ap3 = RepAppl(d1); | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 	f = (Functor) (*ap2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* compare functors */ | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 	if (f != (Functor) *ap3) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  goto cufail; | 
					
						
							|  |  |  | 	if (IsExtensionFunctor(f)) { | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 	  if (unify_extension(f, d0, ap2, d1)) | 
					
						
							|  |  |  | 	    continue; | 
					
						
							|  |  |  | 	  goto cufail; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	/* now link the two structures so that no one else will */ | 
					
						
							|  |  |  | 	/* come here */ | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 	/* store the terms to visit */ | 
					
						
							|  |  |  | 	if (RATIONAL_TREES || pt0 < pt0_end) { | 
					
						
							|  |  |  | 	  to_visit --; | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  | 	  unif++; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	  if ((void *)to_visit < (void *)unif) { | 
					
						
							| 
									
										
										
										
											2009-05-22 21:53:24 -05:00
										 |  |  | 	    CELL **urec = (CELL **)unif; | 
					
						
							| 
									
										
										
										
											2012-03-15 22:37:13 +00:00
										 |  |  | 	    to_visit = (struct v_record *)Yap_shift_visit((CELL **)to_visit, &urec, NULL); | 
					
						
							| 
									
										
										
										
											2009-05-22 21:53:24 -05:00
										 |  |  | 	    unif = (struct unif_record *)urec; | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	  to_visit->start0 = pt0; | 
					
						
							|  |  |  | 	  to_visit->end0 = pt0_end; | 
					
						
							|  |  |  | 	  to_visit->start1 = pt1; | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  | 	  unif[-1].old = *pt0; | 
					
						
							|  |  |  | 	  unif[-1].ptr = pt0; | 
					
						
							|  |  |  | 	  *pt0 = d1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	d0 = ArityOfFunctor(f); | 
					
						
							|  |  |  | 	pt0 = ap2; | 
					
						
							|  |  |  | 	pt0_end = ap2 + d0; | 
					
						
							|  |  |  | 	pt1 = ap3; | 
					
						
							|  |  |  | 	continue; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |       goto cufail; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       derefa_body(d1, ptd1, unify_comp_nvar_unk, unify_comp_nvar_nvar); | 
					
						
							|  |  |  |       /* d1 and pt2 have the unbound value, whereas d0 is bound */ | 
					
						
							| 
									
										
										
										
											2011-03-18 19:34:58 +00:00
										 |  |  |       Bind_Global(ptd1, d0); | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |       if (Yap_rational_tree_loop(ptd1-1, ptd1, (CELL **)to_visit, (CELL **)unif)) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	goto cufail; | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     derefa_body(d0, ptd0, unify_comp_unk, unify_comp_nvar); | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |     /* first arg var */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       register CELL d1; | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |       register CELL *ptd1; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |       ptd1 = pt1; | 
					
						
							|  |  |  |       d1 = ptd1[0]; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* pt2 is unbound */ | 
					
						
							|  |  |  |       deref_head(d1, unify_comp_var_unk); | 
					
						
							|  |  |  |     unify_comp_var_nvar: | 
					
						
							|  |  |  |       /* pt2 is unbound and d1 is bound */ | 
					
						
							| 
									
										
										
										
											2011-03-18 19:34:58 +00:00
										 |  |  |       Bind_Global(ptd0, d1); | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |       if (Yap_rational_tree_loop(ptd0-1, ptd0, (CELL **)to_visit, (CELL **)unif)) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	goto cufail; | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       derefa_body(d1, ptd1, unify_comp_var_unk, unify_comp_var_nvar); | 
					
						
							|  |  |  |       /* ptd0 and ptd1 are unbound */ | 
					
						
							|  |  |  |       UnifyGlobalCells(ptd0, ptd1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* Do we still have compound terms to visit */ | 
					
						
							| 
									
										
										
										
											2006-12-13 16:10:26 +00:00
										 |  |  |   if (to_visit < to_visit_base) { | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  |     pt0 = to_visit->start0; | 
					
						
							|  |  |  |     pt0_end = to_visit->end0; | 
					
						
							|  |  |  |     pt1 = to_visit->start1; | 
					
						
							|  |  |  |     to_visit++; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     goto loop; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  |   /* restore bindigs */ | 
					
						
							|  |  |  |   while (unif-- != unif_base) { | 
					
						
							|  |  |  |     CELL *pt0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pt0 = unif->ptr; | 
					
						
							|  |  |  |     *pt0 = unif->old; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | cufail: | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  |   /* restore bindigs */ | 
					
						
							|  |  |  |   while (unif-- != unif_base) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     CELL *pt0; | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pt0 = unif->ptr; | 
					
						
							|  |  |  |     *pt0 = unif->old; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   return FALSE; | 
					
						
							|  |  |  | #ifdef THREADS
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  | #define Yap_REGS (*Yap_regp)  
 | 
					
						
							|  |  |  | #elif defined(SHADOW_REGS)
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if defined(B) || defined(TR)
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | #undef Yap_REGS
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* defined(B) || defined(TR) */
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-05-22 18:35:24 -05:00
										 |  |  | #undef unif_base
 | 
					
						
							|  |  |  | #undef to_visit_base
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int  | 
					
						
							|  |  |  | OCUnify(register CELL d0, register CELL d1) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   register CELL *pt0, *pt1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if SHADOW_HB
 | 
					
						
							|  |  |  |   register CELL *HBREG = HB; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_head(d0, oc_unify_unk); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | oc_unify_nvar: | 
					
						
							|  |  |  |   /* d0 is bound */ | 
					
						
							|  |  |  |   deref_head(d1, oc_unify_nvar_unk); | 
					
						
							|  |  |  | oc_unify_nvar_nvar: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (d0 == d1) { | 
					
						
							| 
									
										
										
										
											2005-08-17 13:35:52 +00:00
										 |  |  |     return (!rational_tree(d0)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* both arguments are bound */ | 
					
						
							|  |  |  |   if (IsPairTerm(d0)) { | 
					
						
							|  |  |  |     if (!IsPairTerm(d1)) { | 
					
						
							|  |  |  | 	return (FALSE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pt0 = RepPair(d0); | 
					
						
							|  |  |  |     pt1 = RepPair(d1); | 
					
						
							|  |  |  |     return (OCUnify_complex(pt0 - 1, pt0 + 1, pt1 - 1)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (IsApplTerm(d0)) { | 
					
						
							|  |  |  |     if (!IsApplTerm(d1)) | 
					
						
							|  |  |  | 	return (FALSE); | 
					
						
							|  |  |  |     pt0 = RepAppl(d0); | 
					
						
							|  |  |  |     d0 = *pt0; | 
					
						
							|  |  |  |     pt1 = RepAppl(d1); | 
					
						
							|  |  |  |     d1 = *pt1; | 
					
						
							|  |  |  |     if (d0 != d1) { | 
					
						
							|  |  |  |       return (FALSE); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       if (IsExtensionFunctor((Functor)d0)) { | 
					
						
							|  |  |  | 	switch(d0) { | 
					
						
							|  |  |  | 	case (CELL)FunctorDBRef: | 
					
						
							|  |  |  | 	  return(pt0 == pt1); | 
					
						
							|  |  |  | 	case (CELL)FunctorLongInt: | 
					
						
							|  |  |  | 	  return(pt0[1] == pt1[1]); | 
					
						
							|  |  |  | 	case (CELL)FunctorDouble: | 
					
						
							|  |  |  | 	  return(FloatOfTerm(AbsAppl(pt0)) == FloatOfTerm(AbsAppl(pt1))); | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  | 	case (CELL)FunctorString: | 
					
						
							| 
									
										
										
										
											2013-12-04 23:01:30 +00:00
										 |  |  | 	  return(strcmp( (const char *)(pt0+2),  (const char *)(pt1+2)) == 0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  | 	case (CELL)FunctorBigInt: | 
					
						
							| 
									
										
										
										
											2010-05-28 12:07:01 +01:00
										 |  |  | 	  return(Yap_gmp_tcmp_big_big(AbsAppl(pt0),AbsAppl(pt0)) == 0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* USE_GMP */
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 	  return(FALSE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return (OCUnify_complex(pt0, pt0 + ArityOfFunctor((Functor) d0), | 
					
						
							|  |  |  | 			      pt1)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     return(FALSE); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_body(d1, pt1, oc_unify_nvar_unk, oc_unify_nvar_nvar); | 
					
						
							|  |  |  |   /* d0 is bound and d1 is unbound */ | 
					
						
							| 
									
										
										
										
											2014-05-29 11:32:28 +02:00
										 |  |  |   YapBind(pt1, d0); | 
					
						
							| 
									
										
										
										
											2009-05-17 12:39:51 -07:00
										 |  |  |   /* local variables cannot be in a term */ | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   if (pt1 > HR && pt1 < LCL0) | 
					
						
							| 
									
										
										
										
											2009-05-17 12:39:51 -07:00
										 |  |  |     return TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (rational_tree(d0)) | 
					
						
							|  |  |  |     return(FALSE); | 
					
						
							|  |  |  |   return (TRUE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_body(d0, pt0, oc_unify_unk, oc_unify_nvar); | 
					
						
							|  |  |  |   /* pt0 is unbound */ | 
					
						
							|  |  |  |   deref_head(d1, oc_unify_var_unk); | 
					
						
							|  |  |  | oc_unify_var_nvar: | 
					
						
							|  |  |  |   /* pt0 is unbound and d1 is bound */ | 
					
						
							| 
									
										
										
										
											2014-05-29 11:32:28 +02:00
										 |  |  |   YapBind(pt0, d1); | 
					
						
							| 
									
										
										
										
											2009-05-17 12:39:51 -07:00
										 |  |  |   /* local variables cannot be in a term */ | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   if (pt0 > HR && pt0 < LCL0) | 
					
						
							| 
									
										
										
										
											2009-05-17 12:39:51 -07:00
										 |  |  |     return TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (rational_tree(d1)) | 
					
						
							|  |  |  |     return(FALSE); | 
					
						
							|  |  |  |   return (TRUE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_body(d1, pt1, oc_unify_var_unk, oc_unify_var_nvar); | 
					
						
							|  |  |  |   /* d0 and pt1 are unbound */ | 
					
						
							| 
									
										
										
										
											2011-03-18 19:34:58 +00:00
										 |  |  |   UnifyCells(pt0, pt1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   return (TRUE); | 
					
						
							|  |  |  |   return (TRUE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | p_ocunify( USES_REGS1 ) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   return(OCUnify(ARG1,ARG2)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | p_cyclic( USES_REGS1 ) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   if (IsVarTerm(t)) | 
					
						
							|  |  |  |     return(FALSE); | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  |   return rational_tree(t); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-31 04:42:47 -05:00
										 |  |  | bool Yap_IsAcyclicTerm(Term t) | 
					
						
							| 
									
										
										
										
											2012-10-03 09:11:37 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   return !rational_tree(t); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | static Int | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | p_acyclic( USES_REGS1 ) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   if (IsVarTerm(t)) | 
					
						
							|  |  |  |     return(TRUE); | 
					
						
							| 
									
										
										
										
											2005-08-15 14:16:38 +00:00
										 |  |  |   return !rational_tree(t); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int  | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_IUnify(register CELL d0, register CELL d1) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | CACHE_REGS | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #if THREADS
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  |   register REGSTORE *regp = Yap_regp; | 
					
						
							|  |  |  | #define Yap_REGS (*regp)
 | 
					
						
							|  |  |  | #elif SHADOW_REGS
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if defined(B) || defined(TR)
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   register REGSTORE *regp = &Yap_REGS; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | #define Yap_REGS (*regp)
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* defined(B) || defined(TR) */
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if SHADOW_HB
 | 
					
						
							|  |  |  |   register CELL *HBREG = HB; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   register CELL *pt0, *pt1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_head(d0, unify_unk); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unify_nvar: | 
					
						
							|  |  |  |   /* d0 is bound */ | 
					
						
							|  |  |  |   deref_head(d1, unify_nvar_unk); | 
					
						
							|  |  |  | unify_nvar_nvar: | 
					
						
							|  |  |  |   /* both arguments are bound */ | 
					
						
							|  |  |  |   if (d0 == d1) | 
					
						
							| 
									
										
										
										
											2005-02-18 21:34:02 +00:00
										 |  |  |     return TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (IsPairTerm(d0)) { | 
					
						
							|  |  |  |     if (!IsPairTerm(d1)) { | 
					
						
							|  |  |  |       return (FALSE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pt0 = RepPair(d0); | 
					
						
							|  |  |  |     pt1 = RepPair(d1); | 
					
						
							|  |  |  |     return (IUnify_complex(pt0 - 1, pt0 + 1, pt1 - 1)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (IsApplTerm(d0)) { | 
					
						
							|  |  |  |     pt0 = RepAppl(d0); | 
					
						
							|  |  |  |     d0 = *pt0; | 
					
						
							|  |  |  |     if (!IsApplTerm(d1)) | 
					
						
							|  |  |  |       return (FALSE);       | 
					
						
							|  |  |  |     pt1 = RepAppl(d1); | 
					
						
							|  |  |  |     d1 = *pt1; | 
					
						
							|  |  |  |     if (d0 != d1) { | 
					
						
							|  |  |  |       return (FALSE); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       if (IsExtensionFunctor((Functor)d0)) { | 
					
						
							|  |  |  | 	switch(d0) { | 
					
						
							|  |  |  | 	case (CELL)FunctorDBRef: | 
					
						
							|  |  |  | 	  return(pt0 == pt1); | 
					
						
							|  |  |  | 	case (CELL)FunctorLongInt: | 
					
						
							|  |  |  | 	  return(pt0[1] == pt1[1]); | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  | 	case (CELL)FunctorString: | 
					
						
							| 
									
										
										
										
											2013-12-04 23:01:30 +00:00
										 |  |  | 	  return(strcmp( (const char *)(pt0+2),  (const char *)(pt1+2)) == 0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	case (CELL)FunctorDouble: | 
					
						
							|  |  |  | 	  return(FloatOfTerm(AbsAppl(pt0)) == FloatOfTerm(AbsAppl(pt1))); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  | 	case (CELL)FunctorBigInt: | 
					
						
							| 
									
										
										
										
											2010-05-28 12:07:01 +01:00
										 |  |  | 	  return(Yap_gmp_tcmp_big_big(AbsAppl(pt0),AbsAppl(pt0)) == 0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* USE_GMP */
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 	  return(FALSE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return (IUnify_complex(pt0, pt0 + ArityOfFunctor((Functor) d0), | 
					
						
							|  |  |  | 			     pt1)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     return (FALSE); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_body(d1, pt1, unify_nvar_unk, unify_nvar_nvar); | 
					
						
							|  |  |  |   /* d0 is bound and d1 is unbound */ | 
					
						
							| 
									
										
										
										
											2014-05-29 11:32:28 +02:00
										 |  |  |   YapBind(pt1, d0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   return (TRUE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_body(d0, pt0, unify_unk, unify_nvar); | 
					
						
							|  |  |  |   /* pt0 is unbound */ | 
					
						
							|  |  |  |   deref_head(d1, unify_var_unk); | 
					
						
							|  |  |  | unify_var_nvar: | 
					
						
							|  |  |  |   /* pt0 is unbound and d1 is bound */ | 
					
						
							| 
									
										
										
										
											2014-05-29 11:32:28 +02:00
										 |  |  |   YapBind(pt0, d1); | 
					
						
							| 
									
										
										
										
											2005-02-18 21:34:02 +00:00
										 |  |  |   return TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if TRAILING_REQUIRES_BRANCH
 | 
					
						
							|  |  |  | unify_var_nvar_trail: | 
					
						
							|  |  |  |   DO_TRAIL(pt0); | 
					
						
							| 
									
										
										
										
											2005-02-18 21:34:02 +00:00
										 |  |  |   return TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_body(d1, pt1, unify_var_unk, unify_var_nvar); | 
					
						
							|  |  |  |   /* d0 and pt1 are unbound */ | 
					
						
							| 
									
										
										
										
											2011-03-18 19:34:58 +00:00
										 |  |  |   UnifyCells(pt0, pt1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   return (TRUE); | 
					
						
							| 
									
										
										
										
											2011-03-18 19:34:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #if THREADS
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  | #define Yap_REGS (*Yap_regp)  
 | 
					
						
							|  |  |  | #elif SHADOW_REGS
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if defined(B) || defined(TR)
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | #undef Yap_REGS
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* defined(B) || defined(TR) */
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**********************************************************************
 | 
					
						
							|  |  |  |  *                                                                    * | 
					
						
							|  |  |  |  *                 Conversion from Label to Op                        * | 
					
						
							|  |  |  |  *                                                                    * | 
					
						
							|  |  |  |  **********************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if USE_THREADED_CODE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* mask a hash table that allows for fast reverse translation from
 | 
					
						
							|  |  |  |    instruction address to corresponding opcode */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | InitReverseLookupOpcode(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-12-14 11:57:31 +00:00
										 |  |  |   op_entry *opeptr; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   op_numbers i; | 
					
						
							|  |  |  |   /* 2 K should be OK */ | 
					
						
							|  |  |  |   int hash_size_mask = OP_HASH_SIZE-1; | 
					
						
							| 
									
										
										
										
											2007-04-10 22:13:21 +00:00
										 |  |  |   UInt sz = OP_HASH_SIZE*sizeof(struct opcode_tab_entry); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-04-10 22:13:21 +00:00
										 |  |  |   while (OP_RTABLE == NULL) { | 
					
						
							| 
									
										
										
										
											2014-12-14 11:57:31 +00:00
										 |  |  |     if ((OP_RTABLE = (op_entry *)Yap_AllocCodeSpace(sz)) == NULL) { | 
					
						
							| 
									
										
										
										
											2007-04-10 22:13:21 +00:00
										 |  |  |       if (!Yap_growheap(FALSE, sz, NULL)) { | 
					
						
							| 
									
										
										
										
											2015-09-25 10:57:26 +01:00
										 |  |  | 	Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, | 
					
						
							| 
									
										
										
										
											2007-04-10 22:13:21 +00:00
										 |  |  | 		  "Couldn't obtain space for the reverse translation opcode table"); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   memset(OP_RTABLE, 0, sz); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   opeptr = OP_RTABLE; | 
					
						
							|  |  |  |   /* clear up table */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     int j; | 
					
						
							| 
									
										
										
										
											2004-02-05 16:57:02 +00:00
										 |  |  |     for (j=0; j<OP_HASH_SIZE; j++) { | 
					
						
							|  |  |  |       opeptr[j].opc = 0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       opeptr[j].opnum = _Ystop; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   opeptr = OP_RTABLE; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   opeptr[rtable_hash_op(Yap_opcode(_Ystop),hash_size_mask)].opc | 
					
						
							|  |  |  | 	    = Yap_opcode(_Ystop); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* now place entries */ | 
					
						
							|  |  |  |   for (i = _std_top; i > _Ystop; i--) { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     OPCODE opc = Yap_opcode(i); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     int j = rtable_hash_op(opc,hash_size_mask); | 
					
						
							| 
									
										
										
										
											2004-02-05 16:57:02 +00:00
										 |  |  |     while (opeptr[j].opc) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       if (++j > hash_size_mask) | 
					
						
							|  |  |  | 	j = 0;	   | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* clear entry, no conflict */ | 
					
						
							|  |  |  |     opeptr[j].opnum = i; | 
					
						
							|  |  |  |     opeptr[j].opc = opc; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  | #define UnifyAndTrailGlobalCells(a, b)                                \
 | 
					
						
							|  |  |  |      if((a) > (b)) {                                                  \ | 
					
						
							|  |  |  |        *(a) = (CELL)(b);                                              \ | 
					
						
							|  |  |  |         DO_TRAIL((a), (CELL)(b));                                     \ | 
					
						
							|  |  |  |      } else if((a) < (b)){                                            \ | 
					
						
							|  |  |  |        *(b) = (CELL)(a);                                              \ | 
					
						
							|  |  |  |         DO_TRAIL((b), (CELL)(a));                                     \ | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  |      } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int  | 
					
						
							|  |  |  | unifiable_complex(CELL *pt0, CELL *pt0_end, CELL *pt1) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | CACHE_REGS | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  | #ifdef THREADS
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  |   register REGSTORE *regp = Yap_regp; | 
					
						
							|  |  |  | #define Yap_REGS (*regp)
 | 
					
						
							|  |  |  | #elif defined(SHADOW_REGS)
 | 
					
						
							|  |  |  | #if defined(B) || defined(TR)
 | 
					
						
							|  |  |  |   register REGSTORE *regp = &Yap_REGS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define Yap_REGS (*regp)
 | 
					
						
							|  |  |  | #endif /* defined(B) || defined(TR) || defined(HB) */
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef SHADOW_HB
 | 
					
						
							|  |  |  |   register CELL *HBREG = HB; | 
					
						
							|  |  |  | #endif /* SHADOW_HB */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   struct unif_record  *unif = (struct unif_record *)AuxBase; | 
					
						
							|  |  |  |   struct v_record *to_visit  = (struct v_record *)AuxSp; | 
					
						
							|  |  |  | #define unif_base ((struct unif_record *)AuxBase)
 | 
					
						
							|  |  |  | #define to_visit_base ((struct v_record *)AuxSp)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | loop: | 
					
						
							|  |  |  |   while (pt0 < pt0_end) { | 
					
						
							|  |  |  |     register CELL *ptd0 = pt0+1;  | 
					
						
							|  |  |  |     register CELL d0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ++pt1; | 
					
						
							|  |  |  |     pt0 = ptd0; | 
					
						
							|  |  |  |     d0 = *ptd0; | 
					
						
							|  |  |  |     deref_head(d0, unifiable_comp_unk); | 
					
						
							|  |  |  |   unifiable_comp_nvar: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       register CELL *ptd1 = pt1; | 
					
						
							|  |  |  |       register CELL d1 = *ptd1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       deref_head(d1, unifiable_comp_nvar_unk); | 
					
						
							|  |  |  |     unifiable_comp_nvar_nvar: | 
					
						
							|  |  |  |       if (d0 == d1) | 
					
						
							|  |  |  | 	continue; | 
					
						
							|  |  |  |       if (IsPairTerm(d0)) { | 
					
						
							|  |  |  | 	if (!IsPairTerm(d1)) { | 
					
						
							|  |  |  | 	  goto cufail; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* now link the two structures so that no one else will */ | 
					
						
							|  |  |  | 	/* come here */ | 
					
						
							|  |  |  | 	/* store the terms to visit */ | 
					
						
							|  |  |  | 	if (RATIONAL_TREES || pt0 < pt0_end) { | 
					
						
							|  |  |  | 	  to_visit --; | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  | 	  unif++; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	  if ((void *)to_visit < (void *)unif) { | 
					
						
							|  |  |  | 	    CELL **urec = (CELL **)unif; | 
					
						
							| 
									
										
										
										
											2012-03-15 22:37:13 +00:00
										 |  |  | 	    to_visit = (struct v_record *)Yap_shift_visit((CELL **)to_visit, &urec, NULL); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  | 	    unif = (struct unif_record *)urec; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	  to_visit->start0 = pt0; | 
					
						
							|  |  |  | 	  to_visit->end0 = pt0_end; | 
					
						
							|  |  |  | 	  to_visit->start1 = pt1; | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  | 	  unif[-1].old = *pt0; | 
					
						
							|  |  |  | 	  unif[-1].ptr = pt0; | 
					
						
							|  |  |  | 	  *pt0 = d1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	pt0_end = (pt0 = RepPair(d0) - 1) + 2; | 
					
						
							|  |  |  | 	pt1 = RepPair(d1) - 1; | 
					
						
							|  |  |  | 	continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (IsApplTerm(d0)) { | 
					
						
							|  |  |  | 	register Functor f; | 
					
						
							|  |  |  | 	register CELL *ap2, *ap3; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!IsApplTerm(d1)) { | 
					
						
							|  |  |  | 	  goto cufail; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* store the terms to visit */ | 
					
						
							|  |  |  | 	ap2 = RepAppl(d0); | 
					
						
							|  |  |  | 	ap3 = RepAppl(d1); | 
					
						
							|  |  |  | 	f = (Functor) (*ap2); | 
					
						
							|  |  |  | 	/* compare functors */ | 
					
						
							|  |  |  | 	if (f != (Functor) *ap3) | 
					
						
							|  |  |  | 	  goto cufail; | 
					
						
							|  |  |  | 	if (IsExtensionFunctor(f)) { | 
					
						
							|  |  |  | 	  if (unify_extension(f, d0, ap2, d1)) | 
					
						
							|  |  |  | 	    continue; | 
					
						
							|  |  |  | 	  goto cufail; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* now link the two structures so that no one else will */ | 
					
						
							|  |  |  | 	/* come here */ | 
					
						
							|  |  |  | 	/* store the terms to visit */ | 
					
						
							|  |  |  | 	if (RATIONAL_TREES || pt0 < pt0_end) { | 
					
						
							|  |  |  | 	  to_visit --; | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  | 	  unif++; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	  if ((void *)to_visit < (void *)unif) { | 
					
						
							|  |  |  | 	    CELL **urec = (CELL **)unif; | 
					
						
							| 
									
										
										
										
											2012-03-15 22:37:13 +00:00
										 |  |  | 	    to_visit = (struct v_record *)Yap_shift_visit((CELL **)to_visit, &urec, NULL); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  | 	    unif = (struct unif_record *)urec; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	  to_visit->start0 = pt0; | 
					
						
							|  |  |  | 	  to_visit->end0 = pt0_end; | 
					
						
							|  |  |  | 	  to_visit->start1 = pt1; | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  | 	  unif[-1].old = *pt0; | 
					
						
							|  |  |  | 	  unif[-1].ptr = pt0; | 
					
						
							|  |  |  | 	  *pt0 = d1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	d0 = ArityOfFunctor(f); | 
					
						
							|  |  |  | 	pt0 = ap2; | 
					
						
							|  |  |  | 	pt0_end = ap2 + d0; | 
					
						
							|  |  |  | 	pt1 = ap3; | 
					
						
							|  |  |  | 	continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       goto cufail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       derefa_body(d1, ptd1, unifiable_comp_nvar_unk, unifiable_comp_nvar_nvar); | 
					
						
							|  |  |  | 	/* d1 and pt2 have the unbound value, whereas d0 is bound */ | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |       *(ptd1) = d0; | 
					
						
							|  |  |  |       DO_TRAIL(ptd1, d0); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  |       continue; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     derefa_body(d0, ptd0, unifiable_comp_unk, unifiable_comp_nvar); | 
					
						
							|  |  |  |     /* first arg var */ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       register CELL d1; | 
					
						
							|  |  |  |       register CELL *ptd1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ptd1 = pt1; | 
					
						
							|  |  |  |       d1 = ptd1[0]; | 
					
						
							|  |  |  |       /* pt2 is unbound */ | 
					
						
							|  |  |  |       deref_head(d1, unifiable_comp_var_unk); | 
					
						
							|  |  |  |     unifiable_comp_var_nvar: | 
					
						
							|  |  |  |       /* pt2 is unbound and d1 is bound */ | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |       *ptd0 = d1; | 
					
						
							|  |  |  |        DO_TRAIL(ptd0, d1); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  |       continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       derefa_body(d1, ptd1, unifiable_comp_var_unk, unifiable_comp_var_nvar); | 
					
						
							|  |  |  |       /* ptd0 and ptd1 are unbound */ | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |       UnifyAndTrailGlobalCells(ptd0, ptd1); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* Do we still have compound terms to visit */ | 
					
						
							|  |  |  |   if (to_visit < to_visit_base) { | 
					
						
							|  |  |  |     pt0 = to_visit->start0; | 
					
						
							|  |  |  |     pt0_end = to_visit->end0; | 
					
						
							|  |  |  |     pt1 = to_visit->start1; | 
					
						
							|  |  |  |     to_visit++; | 
					
						
							|  |  |  |     goto loop; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  |   /* restore bindigs */ | 
					
						
							|  |  |  |   while (unif-- != unif_base) { | 
					
						
							|  |  |  |     CELL *pt0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pt0 = unif->ptr; | 
					
						
							|  |  |  |     *pt0 = unif->old; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | cufail: | 
					
						
							|  |  |  | #ifdef RATIONAL_TREES
 | 
					
						
							|  |  |  |   /* restore bindigs */ | 
					
						
							|  |  |  |   while (unif-- != unif_base) { | 
					
						
							|  |  |  |     CELL *pt0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pt0 = unif->ptr; | 
					
						
							|  |  |  |     *pt0 = unif->old; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return FALSE; | 
					
						
							|  |  |  | #ifdef THREADS
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  | #define Yap_REGS (*Yap_regp)  
 | 
					
						
							|  |  |  | #elif defined(SHADOW_REGS)
 | 
					
						
							|  |  |  | #if defined(B) || defined(TR)
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  | #endif /* defined(B) || defined(TR) */
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*  don't pollute name space */ | 
					
						
							|  |  |  | #undef to_visit_base
 | 
					
						
							|  |  |  | #undef unif_base
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int  | 
					
						
							|  |  |  | unifiable(CELL d0, CELL d1) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | CACHE_REGS | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  | #if THREADS
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  |   register REGSTORE *regp = Yap_regp; | 
					
						
							|  |  |  | #define Yap_REGS (*regp)
 | 
					
						
							|  |  |  | #elif SHADOW_REGS
 | 
					
						
							|  |  |  | #if defined(B) || defined(TR)
 | 
					
						
							|  |  |  |   register REGSTORE *regp = &Yap_REGS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define Yap_REGS (*regp)
 | 
					
						
							|  |  |  | #endif /* defined(B) || defined(TR) */
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if SHADOW_HB
 | 
					
						
							|  |  |  |   register CELL *HBREG = HB; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   register CELL *pt0, *pt1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_head(d0, unifiable_unk); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unifiable_nvar: | 
					
						
							|  |  |  |   /* d0 is bound */ | 
					
						
							|  |  |  |   deref_head(d1, unifiable_nvar_unk); | 
					
						
							|  |  |  | unifiable_nvar_nvar: | 
					
						
							|  |  |  |   /* both arguments are bound */ | 
					
						
							|  |  |  |   if (d0 == d1) | 
					
						
							|  |  |  |     return TRUE; | 
					
						
							|  |  |  |   if (IsPairTerm(d0)) { | 
					
						
							|  |  |  |     if (!IsPairTerm(d1)) { | 
					
						
							|  |  |  |       return (FALSE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pt0 = RepPair(d0); | 
					
						
							|  |  |  |     pt1 = RepPair(d1); | 
					
						
							|  |  |  |     return (unifiable_complex(pt0 - 1, pt0 + 1, pt1 - 1)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (IsApplTerm(d0)) { | 
					
						
							|  |  |  |     pt0 = RepAppl(d0); | 
					
						
							|  |  |  |     d0 = *pt0; | 
					
						
							|  |  |  |     if (!IsApplTerm(d1)) | 
					
						
							|  |  |  |       return (FALSE);       | 
					
						
							|  |  |  |     pt1 = RepAppl(d1); | 
					
						
							|  |  |  |     d1 = *pt1; | 
					
						
							|  |  |  |     if (d0 != d1) { | 
					
						
							|  |  |  |       return (FALSE); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       if (IsExtensionFunctor((Functor)d0)) { | 
					
						
							|  |  |  | 	switch(d0) { | 
					
						
							|  |  |  | 	case (CELL)FunctorDBRef: | 
					
						
							|  |  |  | 	  return(pt0 == pt1); | 
					
						
							|  |  |  | 	case (CELL)FunctorLongInt: | 
					
						
							|  |  |  | 	  return(pt0[1] == pt1[1]); | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  | 	case (CELL)FunctorString: | 
					
						
							| 
									
										
										
										
											2013-12-04 23:01:30 +00:00
										 |  |  | 	  return(strcmp( (const char *)(pt0+2),  (const char *)(pt1+2)) == 0); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  | 	case (CELL)FunctorDouble: | 
					
						
							|  |  |  | 	  return(FloatOfTerm(AbsAppl(pt0)) == FloatOfTerm(AbsAppl(pt1))); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  | 	case (CELL)FunctorBigInt: | 
					
						
							| 
									
										
										
										
											2010-05-28 12:07:01 +01:00
										 |  |  | 	  return(Yap_gmp_tcmp_big_big(AbsAppl(pt0),AbsAppl(pt0)) == 0); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  | #endif /* USE_GMP */
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 	  return(FALSE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return (unifiable_complex(pt0, pt0 + ArityOfFunctor((Functor) d0), | 
					
						
							|  |  |  | 			     pt1)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     return (FALSE); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_body(d1, pt1, unifiable_nvar_unk, unifiable_nvar_nvar); | 
					
						
							|  |  |  |   /* d0 is bound and d1 is unbound */ | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |   *(pt1) = d0; | 
					
						
							|  |  |  |   DO_TRAIL(pt1, d0); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  |   return (TRUE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_body(d0, pt0, unifiable_unk, unifiable_nvar); | 
					
						
							|  |  |  |   /* pt0 is unbound */ | 
					
						
							|  |  |  |   deref_head(d1, unifiable_var_unk); | 
					
						
							|  |  |  | unifiable_var_nvar: | 
					
						
							|  |  |  |   /* pt0 is unbound and d1 is bound */ | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |   *pt0 = d1; | 
					
						
							|  |  |  |    DO_TRAIL(pt0, d1); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  |   return TRUE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   deref_body(d1, pt1, unifiable_var_unk, unifiable_var_nvar); | 
					
						
							|  |  |  |   /* d0 and pt1 are unbound */ | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |   UnifyAndTrailCells(pt0, pt1); | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  |   return (TRUE); | 
					
						
							|  |  |  | #if THREADS
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  | #define Yap_REGS (*Yap_regp)  
 | 
					
						
							|  |  |  | #elif SHADOW_REGS
 | 
					
						
							|  |  |  | #if defined(B) || defined(TR)
 | 
					
						
							|  |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  | #endif /* defined(B) || defined(TR) */
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | p_unifiable( USES_REGS1 ) | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |   tr_fr_ptr trp, trp0 = TR; | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  |   Term tf = TermNil; | 
					
						
							|  |  |  |   if (!unifiable(ARG1,ARG2)) { | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   trp = TR; | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |   while (trp != trp0) { | 
					
						
							| 
									
										
										
										
											2010-01-21 09:47:55 +00:00
										 |  |  |     Term t[2]; | 
					
						
							|  |  |  |     --trp; | 
					
						
							|  |  |  |     t[0] = TrailTerm(trp); | 
					
						
							|  |  |  |     t[1] = *(CELL *)t[0]; | 
					
						
							|  |  |  |     tf = MkPairTerm(Yap_MkApplTerm(FunctorEq,2,t),tf); | 
					
						
							|  |  |  |     RESET_VARIABLE(t[0]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return Yap_unify(ARG3, tf); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2011-11-03 07:50:34 +09:00
										 |  |  | Yap_Unifiable( Term d0, Term d1 ) | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  | { | 
					
						
							|  |  |  |   CACHE_REGS | 
					
						
							|  |  |  |   tr_fr_ptr trp, trp0 = TR; | 
					
						
							| 
									
										
										
										
											2011-11-03 07:50:34 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |   if (!unifiable(d0,d1)) { | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   trp = TR; | 
					
						
							|  |  |  |   while (trp != trp0) { | 
					
						
							|  |  |  |     Term t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     --trp; | 
					
						
							|  |  |  |     t = TrailTerm(trp); | 
					
						
							|  |  |  |     RESET_VARIABLE(t); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | void  | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_InitUnify(void) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2004-05-13 20:54:58 +00:00
										 |  |  |   Term cm = CurrentModule; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   Yap_InitCPred("unify_with_occurs_check", 2, p_ocunify, SafePredFlag); | 
					
						
							| 
									
										
										
										
											2015-01-04 23:58:23 +00:00
										 |  |  |   /** @pred  unify_with_occurs_check(?T1,?T2) is iso 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Obtain the most general unifier of terms  _T1_ and  _T2_, if there | 
					
						
							|  |  |  | is one. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This predicate implements the full unification algorithm. An example:n | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~{.prolog} | 
					
						
							|  |  |  | unify_with_occurs_check(a(X,b,Z),a(X,A,f(B)). | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | will succeed with the bindings `A = b` and `Z = f(B)`. On the | 
					
						
							|  |  |  | other hand: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~{.prolog} | 
					
						
							|  |  |  | unify_with_occurs_check(a(X,b,Z),a(X,A,f(Z)). | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | would fail, because `Z` is not unifiable with `f(Z)`. Note that | 
					
						
							|  |  |  | `(=)/2` would succeed for the previous examples, giving the following | 
					
						
							|  |  |  | bindings `A = b` and `Z = f(Z)`. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2010-08-04 21:50:19 +01:00
										 |  |  |   Yap_InitCPred("acyclic_term", 1, p_acyclic, SafePredFlag|TestPredFlag); | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | /** @pred  acyclic_term( _T_) is iso 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Succeeds if there are loops in the term  _T_, that is, it is an infinite term. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2004-02-09 14:19:05 +00:00
										 |  |  |   CurrentModule = TERMS_MODULE; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   Yap_InitCPred("cyclic_term", 1, p_cyclic, SafePredFlag|TestPredFlag); | 
					
						
							| 
									
										
										
										
											2011-10-27 12:35:07 +02:00
										 |  |  |   Yap_InitCPred("unifiable", 3, p_unifiable, 0); | 
					
						
							| 
									
										
										
										
											2004-05-13 20:54:58 +00:00
										 |  |  |   CurrentModule = cm; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void  | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_InitAbsmi(void) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-11-05 16:13:47 +00:00
										 |  |  |   /* initialize access to abstract machine instructions */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if USE_THREADED_CODE
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   Yap_absmi(1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   InitReverseLookupOpcode(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-04 14:47:18 +00:00
										 |  |  | void | 
					
						
							|  |  |  | Yap_TrimTrail(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2008-06-05 18:29:52 +00:00
										 |  |  | #ifdef saveregs
 | 
					
						
							|  |  |  | #undef saveregs
 | 
					
						
							|  |  |  | #define saveregs()
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #ifdef setregs
 | 
					
						
							|  |  |  | #undef setregs
 | 
					
						
							|  |  |  | #define setregs()
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-06-15 13:51:30 -05:00
										 |  |  | #if SHADOW_HB
 | 
					
						
							|  |  |  |   register CELL *HBREG = HB; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-04 14:47:18 +00:00
										 |  |  | #include "trim_trail.h"
 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-01-04 23:58:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | //! @}
 |