| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | /*************************************************************************
 | 
					
						
							|  |  |  | *									 * | 
					
						
							| 
									
										
										
										
											2015-01-30 07:25:34 +00:00
										 |  |  | *	 Yap Prolog 							 * | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | *									 * | 
					
						
							|  |  |  | *	Yap Prolog was developed at NCCUP - Universidade do Porto	 * | 
					
						
							| 
									
										
										
										
											2015-04-13 13:28:17 +01:00
										 |  |  | *	\z								 * | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | ************************************************************************** | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * File:		absmi.c							 * | 
					
						
							|  |  |  | * comments:	Portable abstract machine interpreter                    * | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | * Last rev:     $Date: 2008-08-13 01:16:26 $,$Author: vsc $ | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2004-03-10 14:59:55 +00:00
										 |  |  | * $Log: not supported by cvs2svn $ | 
					
						
							| 
									
										
										
										
											2008-08-13 01:16:26 +00:00
										 |  |  | * Revision 1.246  2008/08/12 01:27:22  vsc | 
					
						
							| 
									
										
										
										
											2004-03-10 14:59:55 +00:00
										 |  |  | *									 * | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | *									 * | 
					
						
							|  |  |  | *************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | /**
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @file absmi.c | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 18:50:04 -05:00
										 |  |  | @defgroup Efficiency Efficiency Considerations | 
					
						
							|  |  |  | @ingroup YAPProgramming | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | We next discuss several issues on trying to make Prolog programs run | 
					
						
							|  |  |  | fast in YAP. We assume two different programming styles: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-06 18:11:52 +00:00
										 |  |  | + Execution of <em>deterministic</em> programs ofte | 
					
						
							|  |  |  | n | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | boils down to a recursive loop of the form: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | loop(Env) :- | 
					
						
							|  |  |  |         do_something(Env,NewEnv), | 
					
						
							|  |  |  |         loop(NewEnv). | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #define IN_ABSMI_C 1
 | 
					
						
							| 
									
										
										
										
											2015-08-07 16:57:53 -05:00
										 |  |  | #define _INATIVE 1
 | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | #define HAS_CACHE_REGS 1
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "absmi.h"
 | 
					
						
							|  |  |  | #include "heapgc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-18 18:52:41 +00:00
										 |  |  | #include "cut_c.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-06 18:11:52 +00:00
										 |  |  | #if YAP_JIT
 | 
					
						
							|  |  |  | #include "IsGround.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-18 10:03:57 +00:00
										 |  |  | TraceContext **curtrace; | 
					
						
							|  |  |  | yamop *curpreg; | 
					
						
							|  |  |  | BlocksContext **globalcurblock; | 
					
						
							|  |  |  | COUNT ineedredefinedest; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | yamop *headoftrace; | 
					
						
							| 
									
										
										
										
											2015-02-18 10:03:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-06 18:11:52 +00:00
										 |  |  | NativeContext *NativeArea; | 
					
						
							|  |  |  | IntermediatecodeContext *IntermediatecodeArea; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CELL l; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CELL nnexec; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-18 10:03:57 +00:00
										 |  |  | Environment *Yap_ExpEnvP, Yap_ExpEnv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void **Yap_ABSMI_ControlLabels; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static Int traced_absmi(void) { return Yap_traced_absmi(); } | 
					
						
							| 
									
										
										
										
											2015-02-06 18:11:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-30 11:18:35 +01:00
										 |  |  | #ifndef YREG
 | 
					
						
							|  |  |  | #define YREG YENV
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-18 10:03:57 +00:00
										 |  |  | void **Yap_ABSMI_OPCODES; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-12 01:27:23 +00:00
										 |  |  | #ifdef PUSH_X
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* keep X as a global variable */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | Term Yap_XREGS[MaxTemps]; /* 29                                     */ | 
					
						
							| 
									
										
										
										
											2008-08-12 01:27:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #include "arith2.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-18 03:00:19 +00:00
										 |  |  | // #include "print_preg.h"
 | 
					
						
							|  |  |  | //#include "sprint_op.hpp"
 | 
					
						
							|  |  |  | //#include "print_op.hpp"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef COROUTINING
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   Imagine we are interrupting the execution, say, because we have a spy | 
					
						
							|  |  |  |    point or because we have goals to wake up. This routine saves the current | 
					
						
							|  |  |  |    live temporary registers into a structure pointed to by register ARG1. | 
					
						
							|  |  |  |    The registers are then recovered by a nasty builtin | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |    called | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static Term push_live_regs(yamop *pco) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |   CELL *lab = (CELL *)(pco->y_u.l.l); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   CELL max = lab[0]; | 
					
						
							| 
									
										
										
										
											2002-02-04 16:12:54 +00:00
										 |  |  |   CELL curr = lab[1]; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   Term tp = MkIntegerTerm((Int)pco); | 
					
						
							|  |  |  |   Term tcp = MkIntegerTerm((Int)CP); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   Term tenv = MkIntegerTerm((Int)(LCL0 - ENV)); | 
					
						
							|  |  |  |   Term tyenv = MkIntegerTerm((Int)(LCL0 - YENV)); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   CELL *start = HR; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   Int tot = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   HR++; | 
					
						
							|  |  |  |   *HR++ = tp; | 
					
						
							|  |  |  |   *HR++ = tcp; | 
					
						
							|  |  |  |   *HR++ = tenv; | 
					
						
							|  |  |  |   *HR++ = tyenv; | 
					
						
							|  |  |  |   tot += 4; | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     CELL i; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     lab += 2; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     for (i = 0; i <= max; i++) { | 
					
						
							|  |  |  |       if (i == 8 * CellSize) { | 
					
						
							|  |  |  |         curr = lab[0]; | 
					
						
							|  |  |  |         lab++; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       if (curr & 1) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         CELL d1; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         tot += 2; | 
					
						
							|  |  |  |         HR[0] = MkIntTerm(i); | 
					
						
							|  |  |  |         d1 = XREGS[i]; | 
					
						
							|  |  |  |         deref_head(d1, wake_up_unk); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       wake_up_nonvar: | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         /* just copy it to the heap */ | 
					
						
							|  |  |  |         HR[1] = d1; | 
					
						
							|  |  |  |         HR += 2; | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           CELL *pt0; | 
					
						
							|  |  |  |           deref_body(d1, pt0, wake_up_unk, wake_up_nonvar); | 
					
						
							|  |  |  |           /* bind it, in case it is a local variable */ | 
					
						
							|  |  |  |           if (pt0 <= HR) { | 
					
						
							|  |  |  |             /* variable is safe */ | 
					
						
							|  |  |  |             HR[1] = (CELL)pt0; | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             d1 = Unsigned(HR + 1); | 
					
						
							|  |  |  |             RESET_VARIABLE(HR + 1); | 
					
						
							|  |  |  |             Bind_Local(pt0, d1); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         HR += 2; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       curr >>= 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     start[0] = (CELL)Yap_MkFunctor(AtomTrue, tot); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     return (AbsAppl(start)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-28 19:32:55 +00:00
										 |  |  | #if USE_THREADED_CODE && (defined(ANALYST) || defined(DEBUG))
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | char *Yap_op_names[] = { | 
					
						
							|  |  |  | #define OPCODE(OP, TYPE) #OP
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | #include "YapOpcodes.h"
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | #undef OPCODE
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int check_alarm_fail_int(int CONT USES_REGS) { | 
					
						
							|  |  |  | #if defined(_MSC_VER) || defined(__MINGW32__)
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   /* I need this for Windows and any system where SIGINT
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |      is not proceesed by same thread as absmi */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (LOCAL_PrologMode & (AbortMode | InterruptMode)) { | 
					
						
							|  |  |  |     CalculateStackGap(PASS_REGS1); | 
					
						
							|  |  |  |     return CONT; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (Yap_get_signal(YAP_FAIL_SIGNAL)) { | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-03-06 02:09:48 +00:00
										 |  |  |   if (!Yap_has_a_signal()) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     /* no need to look into GC */ | 
					
						
							|  |  |  |     CalculateStackGap(PASS_REGS1); | 
					
						
							| 
									
										
										
										
											2014-03-06 02:09:48 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   // fail even if there are more signals, they will have to be dealt later.
 | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |   return -1; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int stack_overflow(PredEntry *pe, CELL *env, yamop *cp, | 
					
						
							|  |  |  |                           arity_t nargs USES_REGS) { | 
					
						
							| 
									
										
										
										
											2016-04-10 06:21:17 -07:00
										 |  |  |   if (Unsigned(YREG) - Unsigned(HR) < StackGap(PASS_REGS1) || | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       Yap_get_signal(YAP_STOVF_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |     S = (CELL *)pe; | 
					
						
							| 
									
										
										
										
											2015-08-07 16:57:53 -05:00
										 |  |  |     if (!Yap_locked_gc(nargs, env, cp)) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       Yap_NilError(RESOURCE_ERROR_STACK, LOCAL_ErrorMessage); | 
					
						
							| 
									
										
										
										
											2014-03-11 15:33:38 +00:00
										 |  |  |       return 0; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int code_overflow(CELL *yenv USES_REGS) { | 
					
						
							|  |  |  |   if (Yap_get_signal(YAP_CDOVF_SIGNAL)) { | 
					
						
							|  |  |  |     CELL cut_b = LCL0 - (CELL *)(yenv[E_CB]); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* do a garbage collection first to check if we can recover memory */ | 
					
						
							| 
									
										
										
										
											2014-09-09 23:50:43 -05:00
										 |  |  |     if (!Yap_locked_growheap(false, 0, NULL)) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       Yap_NilError(RESOURCE_ERROR_HEAP, "YAP failed to grow heap: %s", | 
					
						
							|  |  |  |                    LOCAL_ErrorMessage); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |       return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     CACHE_A1(); | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |     if (yenv == ASP) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       yenv[E_CB] = (CELL)(LCL0 - cut_b); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_handler(PredEntry *pe USES_REGS) { | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-13 08:42:57 +00:00
										 |  |  |   //  printf("D %lx %p\n", LOCAL_ActiveSignals, P);
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   /* tell whether we can creep or not, this is hard because we will
 | 
					
						
							|  |  |  |      lose the info RSN | 
					
						
							|  |  |  |   */ | 
					
						
							|  |  |  |   BEGD(d0); | 
					
						
							|  |  |  |   d0 = pe->ArityOfPE; | 
					
						
							|  |  |  |   if (d0 == 0) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     HR[1] = MkAtomTerm((Atom)pe->FunctorOfPred); | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |     HR[d0 + 2] = AbsAppl(HR); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     *HR = (CELL)pe->FunctorOfPred; | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |     HR++; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     BEGP(pt1); | 
					
						
							|  |  |  |     pt1 = XREGS + 1; | 
					
						
							|  |  |  |     for (; d0 > 0; --d0) { | 
					
						
							|  |  |  |       BEGD(d1); | 
					
						
							|  |  |  |       BEGP(pt0); | 
					
						
							|  |  |  |       pt0 = pt1; | 
					
						
							|  |  |  |       d1 = *pt0; | 
					
						
							|  |  |  |       deref_head(d1, creep_unk); | 
					
						
							|  |  |  |     creep_nonvar: | 
					
						
							|  |  |  |       /* just copy it to the heap */ | 
					
						
							|  |  |  |       pt1++; | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |       *HR++ = d1; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |       continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       derefa_body(d1, pt0, creep_unk, creep_nonvar); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |       if (pt0 <= HR) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         /* variable is safe */ | 
					
						
							|  |  |  |         *HR++ = (CELL)pt0; | 
					
						
							|  |  |  |         pt1++; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         /* bind it, in case it is a local variable */ | 
					
						
							|  |  |  |         d1 = Unsigned(HR); | 
					
						
							|  |  |  |         RESET_VARIABLE(HR); | 
					
						
							|  |  |  |         pt1++; | 
					
						
							|  |  |  |         HR += 1; | 
					
						
							|  |  |  |         Bind_Local(pt0, d1); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       ENDP(pt0); | 
					
						
							|  |  |  |       ENDD(d1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ENDP(pt1); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   ENDD(d0); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   HR[0] = Yap_Module_Name(pe); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   ARG1 = (Term)AbsPair(HR); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   HR += 2; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #ifdef COROUTINING
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (Yap_get_signal(YAP_WAKEUP_SIGNAL)) { | 
					
						
							|  |  |  |     CalculateStackGap(PASS_REGS1); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     ARG2 = Yap_ListOfWokenGoals(); | 
					
						
							|  |  |  |     pe = WakeUpCode; | 
					
						
							|  |  |  |     /* no more goals to wake up */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     Yap_UpdateTimedVar(LOCAL_WokenGoals, TermNil); | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   } else | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   { | 
					
						
							|  |  |  |     CalculateStackGap(PASS_REGS1); | 
					
						
							|  |  |  |     pe = CreepCode; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   P = pe->CodeOfPred; | 
					
						
							|  |  |  | #ifdef LOW_LEVEL_TRACER
 | 
					
						
							|  |  |  |   if (Yap_do_low_level_trace) | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     low_level_trace(enter_pred, pe, XREGS + 1); | 
					
						
							|  |  |  | #endif /* LOW_LEVEL_TRACE */
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   /* for profiler */ | 
					
						
							|  |  |  |   CACHE_A1(); | 
					
						
							| 
									
										
										
										
											2014-09-09 23:50:43 -05:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | // interrupt handling code that sets up the case when we do not have
 | 
					
						
							|  |  |  | // a guaranteed environment.
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int safe_interrupt_handler(PredEntry *pe USES_REGS) { | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   CELL *npt = HR; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //  printf("D %lx %p\n", LOCAL_ActiveSignals, P);
 | 
					
						
							|  |  |  |   /* tell whether we can creep or not, this is hard because we will
 | 
					
						
							|  |  |  |      lose the info RSN | 
					
						
							|  |  |  |   */ | 
					
						
							|  |  |  |   BEGD(d0); | 
					
						
							|  |  |  |   S = (CELL *)pe; | 
					
						
							|  |  |  |   d0 = pe->ArityOfPE; | 
					
						
							|  |  |  |   if (d0 == 0) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     HR[1] = MkAtomTerm((Atom)pe->FunctorOfPred); | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |     HR[d0 + 2] = AbsAppl(HR); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     HR += d0 + 1 + 2; | 
					
						
							|  |  |  |     *npt++ = (CELL)pe->FunctorOfPred; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |     BEGP(pt1); | 
					
						
							|  |  |  |     pt1 = XREGS + 1; | 
					
						
							|  |  |  |     for (; d0 > 0; --d0) { | 
					
						
							|  |  |  |       BEGD(d1); | 
					
						
							|  |  |  |       d1 = *pt1; | 
					
						
							|  |  |  |     loop: | 
					
						
							|  |  |  |       if (!IsVarTerm(d1)) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         /* just copy it to the heap */ | 
					
						
							|  |  |  |         pt1++; | 
					
						
							|  |  |  |         *npt++ = d1; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         if (VarOfTerm(d1) < H0 || VarOfTerm(d1) > HR) { | 
					
						
							|  |  |  |           d1 = Deref(d1); | 
					
						
							|  |  |  |           if (VarOfTerm(d1) < H0 || VarOfTerm(d1) > HR) { | 
					
						
							|  |  |  |             Term v = MkVarTerm(); | 
					
						
							|  |  |  |             YapBind(VarOfTerm(d1), v); | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             goto loop; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           *npt++ = d1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |       ENDD(d1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ENDP(pt1); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   ENDD(d0); | 
					
						
							|  |  |  |   npt[0] = Yap_Module_Name(pe); | 
					
						
							|  |  |  |   ARG1 = AbsPair(npt); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   HR += 2; | 
					
						
							|  |  |  | #ifdef COROUTINING
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (Yap_get_signal(YAP_WAKEUP_SIGNAL)) { | 
					
						
							|  |  |  |     CalculateStackGap(PASS_REGS1); | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |     ARG2 = Yap_ListOfWokenGoals(); | 
					
						
							|  |  |  |     pe = WakeUpCode; | 
					
						
							|  |  |  |     /* no more goals to wake up */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     Yap_UpdateTimedVar(LOCAL_WokenGoals, TermNil); | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   } else | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   { | 
					
						
							|  |  |  |     CalculateStackGap(PASS_REGS1); | 
					
						
							|  |  |  |     pe = CreepCode; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-09-02 14:19:23 -05:00
										 |  |  |   // allocate and fill out an environment
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   YENV = ASP; | 
					
						
							|  |  |  |   CACHE_Y_AS_ENV(YREG); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   ENV_YREG[E_CP] = (CELL)CP; | 
					
						
							|  |  |  |   ENV_YREG[E_E] = (CELL)ENV; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | #ifdef DEPTH_LIMIT
 | 
					
						
							|  |  |  |   ENV_YREG[E_DEPTH] = DEPTH; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | #endif /* DEPTH_LIMIT */
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   ENV = ENV_YREG; | 
					
						
							|  |  |  |   ENV_YREG = (CELL *)((CELL)ENV_YREG + ENV_Size(CP)); | 
					
						
							|  |  |  |   WRITEBACK_Y_AS_ENV(); | 
					
						
							|  |  |  |   ENDCACHE_Y_AS_ENV(); | 
					
						
							|  |  |  |   CP = P; | 
					
						
							|  |  |  |   P = pe->CodeOfPred; | 
					
						
							|  |  |  | #ifdef DEPTH_LIMIT
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (DEPTH <= MkIntTerm(1)) { /* I assume Module==0 is primitives */ | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |     if (pe->ModuleOfPred) { | 
					
						
							|  |  |  |       if (DEPTH == MkIntTerm(0)) | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         DEPTH = RESET_DEPTH(); | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |   } else if (pe->ModuleOfPred) { | 
					
						
							|  |  |  |     DEPTH -= MkIntConstant(2); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | #endif /* DEPTH_LIMIT */
 | 
					
						
							| 
									
										
										
										
											2014-09-09 23:50:43 -05:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_handlerc(PredEntry *pe USES_REGS) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   /* do creep in call                                     */ | 
					
						
							|  |  |  |   ENV = YENV; | 
					
						
							|  |  |  |   CP = NEXTOP(P, Osbpp); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   YENV = (CELL *)(((char *)YENV) + P->y_u.Osbpp.s); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #ifdef FROZEN_STACKS
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     choiceptr top_b = PROTECT_FROZEN_B(B); | 
					
						
							|  |  |  | #ifdef YAPOR_SBA
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (YENV > (CELL *)top_b || YENV < HR) | 
					
						
							|  |  |  |       YENV = (CELL *)top_b; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (YENV > (CELL *)top_b) | 
					
						
							|  |  |  |       YENV = (CELL *)top_b; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #endif /* YAPOR_SBA */
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     else | 
					
						
							|  |  |  |       YENV = YENV + ENV_Size(CP); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (YENV > (CELL *)B) | 
					
						
							|  |  |  |     YENV = (CELL *)B; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   else | 
					
						
							|  |  |  |     /* I am not sure about this */ | 
					
						
							|  |  |  |     YENV = YENV + ENV_Size(CP); | 
					
						
							|  |  |  | #endif /* FROZEN_STACKS */
 | 
					
						
							|  |  |  |   /* setup GB */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   YENV[E_CB] = (CELL)B; | 
					
						
							|  |  |  |   return interrupt_handler(pe PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_handler_either(Term t_cut, PredEntry *pe USES_REGS) { | 
					
						
							|  |  |  |   int rc; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   ARG1 = push_live_regs(NEXTOP(P, Osbpp)); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #ifdef FROZEN_STACKS
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     choiceptr top_b = PROTECT_FROZEN_B(B); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | // protect registers before we mess about.
 | 
					
						
							|  |  |  | // recompute YENV and get ASP
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #ifdef YAPOR_SBA
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (YENV > (CELL *)top_b || YENV < HR) | 
					
						
							|  |  |  |       YENV = (CELL *)top_b; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (YENV > (CELL *)top_b) | 
					
						
							|  |  |  |       YENV = (CELL *)top_b; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #endif /* YAPOR_SBA */
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     else | 
					
						
							|  |  |  |       YENV = YENV + ENV_Size(CP); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (YENV > (CELL *)B) | 
					
						
							|  |  |  |     YENV = (CELL *)B; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #endif /* FROZEN_STACKS */
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   P = NEXTOP(P, Osbpp); | 
					
						
							|  |  |  |   // should we cut? If t_cut == INT(0) no
 | 
					
						
							|  |  |  |   ARG2 = t_cut; | 
					
						
							|  |  |  |   // ASP
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   SET_ASP(YENV, E_CB * sizeof(CELL)); | 
					
						
							|  |  |  |   // do the work.
 | 
					
						
							|  |  |  |   rc = safe_interrupt_handler(pe PASS_REGS); | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   return rc; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | /* to trace interrupt calls */ | 
					
						
							|  |  |  | // #define DEBUG_INTERRUPTS 1
 | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-15 08:35:44 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2014-09-09 23:50:43 -05:00
										 |  |  | static int trace_interrupts = true; | 
					
						
							| 
									
										
										
										
											2013-12-15 08:35:44 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_fail(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d:  (YENV=%p ENV=%p ASP=%p)\n", | 
					
						
							|  |  |  |             worker_id, LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, | 
					
						
							|  |  |  |             __FUNCTION__, __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   check_alarm_fail_int(false PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   /* don't do debugging and stack expansion here: space will
 | 
					
						
							|  |  |  |      be recovered. automatically by fail, so | 
					
						
							|  |  |  |      better wait. | 
					
						
							|  |  |  |   */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (Yap_has_signal(YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2014-09-09 23:50:43 -05:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2014-09-08 17:12:05 -05:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (Yap_has_signal(YAP_CDOVF_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2014-09-09 23:50:43 -05:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* make sure we have the correct environment for continuation */ | 
					
						
							|  |  |  |   ENV = B->cp_env; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   YENV = (CELL *)B; | 
					
						
							|  |  |  |   return interrupt_handler(RepPredProp(Yap_GetPredPropByAtom(AtomFail, 0)) | 
					
						
							|  |  |  |                                PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_execute(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d: (YENV=%p ENV=%p ASP=%p)\n", worker_id, | 
					
						
							|  |  |  |             LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, __FUNCTION__, | 
					
						
							|  |  |  |             __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(true PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (PP) | 
					
						
							|  |  |  |     UNLOCKPE(1, PP); | 
					
						
							|  |  |  |   PP = P->y_u.pp.p0; | 
					
						
							|  |  |  |   if ((P->y_u.pp.p->PredFlags & (NoTracePredFlag | HiddenPredFlag)) && | 
					
						
							|  |  |  |       Yap_only_has_signal(YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     return 2; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   SET_ASP(YENV, E_CB * sizeof(CELL)); | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   if ((v = code_overflow(YENV PASS_REGS)) >= 0) { | 
					
						
							|  |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = stack_overflow(P->y_u.pp.p, ENV, CP, | 
					
						
							|  |  |  |                           P->y_u.pp.p->ArityOfPE PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   return interrupt_handler(P->y_u.pp.p PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_call(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d:  (YENV=%p ENV=%p ASP=%p)\n", | 
					
						
							|  |  |  |             worker_id, LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, | 
					
						
							|  |  |  |             __FUNCTION__, __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(true PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (PP) | 
					
						
							|  |  |  |     UNLOCKPE(1, PP); | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |   PP = P->y_u.Osbpp.p0; | 
					
						
							| 
									
										
										
										
											2014-03-06 02:09:48 +00:00
										 |  |  |   if (Yap_only_has_signal(YAP_CREEP_SIGNAL) && | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       (P->y_u.Osbpp.p->PredFlags & (NoTracePredFlag | HiddenPredFlag))) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     return 2; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |   SET_ASP(YENV, P->y_u.Osbpp.s); | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   if ((v = code_overflow(YENV PASS_REGS)) >= 0) { | 
					
						
							|  |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = stack_overflow(P->y_u.Osbpp.p, YENV, NEXTOP(P, Osbpp), | 
					
						
							|  |  |  |                           P->y_u.Osbpp.p->ArityOfPE PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   return interrupt_handlerc(P->y_u.Osbpp.p PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_pexecute(PredEntry *pen USES_REGS) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   int v; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d:  (YENV=%p ENV=%p ASP=%p)\n", | 
					
						
							|  |  |  |             worker_id, LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, | 
					
						
							|  |  |  |             __FUNCTION__, __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(2 PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (PP) | 
					
						
							|  |  |  |     UNLOCKPE(1, PP); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   PP = NULL; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   if (Yap_only_has_signal(YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2013-12-13 08:42:57 +00:00
										 |  |  |     return 2; /* keep on creeping */ | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   SET_ASP(YENV, E_CB * sizeof(CELL)); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   /* setup GB */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   YENV[E_CB] = (CELL)B; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   if ((v = code_overflow(YENV PASS_REGS)) >= 0) { | 
					
						
							|  |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = stack_overflow(pen, ENV, NEXTOP(P, Osbmp), | 
					
						
							|  |  |  |                           pen->ArityOfPE PASS_REGS)) >= 0) { | 
					
						
							|  |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   CP = NEXTOP(P, Osbmp); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   return interrupt_handler(pen PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static void execute_dealloc(USES_REGS1) { | 
					
						
							|  |  |  |   /* other instructions do depend on S being set by deallocate
 | 
					
						
							|  |  |  |       */ | 
					
						
							| 
									
										
										
										
											2016-02-28 19:32:55 +00:00
										 |  |  |   CELL *ENVYREG = YENV; | 
					
						
							|  |  |  |   S = ENVYREG; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   CP = (yamop *)ENVYREG[E_CP]; | 
					
						
							|  |  |  |   ENV = ENVYREG = (CELL *)ENVYREG[E_E]; | 
					
						
							| 
									
										
										
										
											2016-01-20 22:15:23 +00:00
										 |  |  | #ifdef DEPTH_LIMIT
 | 
					
						
							| 
									
										
										
										
											2016-02-28 19:32:55 +00:00
										 |  |  |   DEPTH = ENVYREG[E_DEPTH]; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | #endif /* DEPTH_LIMIT */
 | 
					
						
							| 
									
										
										
										
											2016-01-20 22:15:23 +00:00
										 |  |  | #ifdef FROZEN_STACKS
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     choiceptr top_b = PROTECT_FROZEN_B(B); | 
					
						
							|  |  |  | #ifdef YAPOR_SBA
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (ENVYREG > (CELL *)top_b || ENVYREG < HR) | 
					
						
							|  |  |  |       ENVYREG = (CELL *)top_b; | 
					
						
							| 
									
										
										
										
											2016-01-20 22:15:23 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (ENVYREG > (CELL *)top_b) | 
					
						
							|  |  |  |       ENVYREG = (CELL *)top_b; | 
					
						
							| 
									
										
										
										
											2016-01-20 22:15:23 +00:00
										 |  |  | #endif /* YAPOR_SBA */
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     else | 
					
						
							|  |  |  |       ENVYREG = (CELL *)((CELL)ENVYREG + ENV_Size(CP)); | 
					
						
							| 
									
										
										
										
											2016-01-20 22:15:23 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (ENVYREG > (CELL *)B) | 
					
						
							|  |  |  |     ENVYREG = (CELL *)B; | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     ENV_YREG = (CELL *)((CELL)ENV_YREG + ENV_Size(CP)); | 
					
						
							| 
									
										
										
										
											2016-01-20 22:15:23 +00:00
										 |  |  | #endif /* FROZEN_STACKS */
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   YENV = ENVYREG; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   P = NEXTOP(P, p); | 
					
						
							| 
									
										
										
										
											2016-01-20 22:15:23 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | /* don't forget I cannot creep at deallocate (where to?) */ | 
					
						
							|  |  |  | /* also, this is unusual in that I have already done deallocate,
 | 
					
						
							|  |  |  |    so I don't need to redo it. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int interrupt_deallocate(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d (YENV=%p ENV=%p ASP=%p)\n", worker_id, | 
					
						
							|  |  |  |             LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, __FUNCTION__, | 
					
						
							|  |  |  |             __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(true PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   /*
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |      don't do a creep here; also, if our instruction is followed by | 
					
						
							|  |  |  |      a execute_c, just wait a bit more */ | 
					
						
							| 
									
										
										
										
											2016-03-30 01:24:34 +01:00
										 |  |  |   if (Yap_only_has_signals(YAP_CREEP_SIGNAL, YAP_WAKEUP_SIGNAL) || | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       /* keep on going if there is something else */ | 
					
						
							|  |  |  |       (P->opc != Yap_opcode(_procceed) && P->opc != Yap_opcode(_cut_e))) { | 
					
						
							|  |  |  |     execute_dealloc(PASS_REGS1); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     return 1; | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     CELL cut_b = LCL0 - (CELL *)(S[E_CB]); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (PP) | 
					
						
							|  |  |  |       UNLOCKPE(1, PP); | 
					
						
							|  |  |  |     PP = PREVOP(P, p)->y_u.p.p; | 
					
						
							|  |  |  |     ASP = YENV + E_CB; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     /* cut_e */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     SET_ASP(YENV, E_CB * sizeof(CELL)); | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |     if ((v = code_overflow(YENV PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |       return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-06 02:09:48 +00:00
										 |  |  |     if (Yap_has_a_signal()) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       PredEntry *pe; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |       if (Yap_op_from_opcode(P->opc) == _cut_e) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         /* followed by a cut */ | 
					
						
							|  |  |  |         ARG1 = MkIntegerTerm(LCL0 - (CELL *)S[E_CB]); | 
					
						
							|  |  |  |         pe = RepPredProp(Yap_GetPredPropByFunc(FunctorCutBy, 1)); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         pe = RepPredProp(Yap_GetPredPropByAtom(AtomTrue, 0)); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-03-30 01:24:34 +01:00
										 |  |  |       // deallocate moves P one step forward.
 | 
					
						
							|  |  |  |       bool rc = interrupt_handler(pe PASS_REGS); | 
					
						
							|  |  |  |       P = NEXTOP(P,p); | 
					
						
							|  |  |  |       return rc; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-01-06 12:30:33 +00:00
										 |  |  |     if (!Yap_locked_gc(0, ENV, YESCODE)) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       Yap_NilError(RESOURCE_ERROR_STACK, LOCAL_ErrorMessage); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     S = ASP; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     S[E_CB] = (CELL)(LCL0 - cut_b); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_cut(USES_REGS1) { | 
					
						
							|  |  |  |   Term t_cut = MkIntegerTerm(LCL0 - (CELL *)YENV[E_CB]); | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d (YENV=%p ENV=%p ASP=%p)\n", worker_id, | 
					
						
							|  |  |  |             LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, __FUNCTION__, | 
					
						
							|  |  |  |             __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(2 PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (!Yap_has_a_signal() || | 
					
						
							|  |  |  |       Yap_only_has_signals(YAP_CDOVF_SIGNAL, YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     return 2; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* find something to fool S */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   P = NEXTOP(P, s); | 
					
						
							|  |  |  |   return interrupt_handler_either(t_cut, PredRestoreRegs PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_cut_t(USES_REGS1) { | 
					
						
							|  |  |  |   Term t_cut = MkIntegerTerm(LCL0 - (CELL *)YENV[E_CB]); | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d: (YENV=%p ENV=%p ASP=%p)\n", worker_id, | 
					
						
							|  |  |  |             LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, __FUNCTION__, | 
					
						
							|  |  |  |             __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(2 PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (!Yap_has_a_signal() || | 
					
						
							|  |  |  |       Yap_only_has_signals(YAP_CDOVF_SIGNAL, YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     return 2; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* find something to fool S */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   P = NEXTOP(P, s); | 
					
						
							|  |  |  |   return interrupt_handler_either(t_cut, PredRestoreRegs PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_cut_e(USES_REGS1) { | 
					
						
							|  |  |  |   Term t_cut = MkIntegerTerm(LCL0 - (CELL *)S[E_CB]); | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d: (YENV=%p ENV=%p ASP=%p)\n", worker_id, | 
					
						
							|  |  |  |             LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, __FUNCTION__, | 
					
						
							|  |  |  |             __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(2 PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (!Yap_only_has_signals(YAP_CDOVF_SIGNAL, YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2014-03-06 02:09:48 +00:00
										 |  |  |     return 2; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   /* find something to fool S */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   P = NEXTOP(P, s); | 
					
						
							|  |  |  |   return interrupt_handler_either(t_cut, PredRestoreRegs PASS_REGS); | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_commit_y(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |   Term t_cut = YENV[P->y_u.yps.y]; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							|  |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d: (YENV=%p ENV=%p ASP=%p)\n", worker_id, | 
					
						
							|  |  |  |             LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, __FUNCTION__, | 
					
						
							|  |  |  |             __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(2 PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (!Yap_has_a_signal() || | 
					
						
							|  |  |  |       Yap_only_has_signals(YAP_CDOVF_SIGNAL, YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |     return 2; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* find something to fool S */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   P = NEXTOP(P, yps); | 
					
						
							|  |  |  |   return interrupt_handler_either(t_cut, PredRestoreRegs PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_commit_x(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |   Term t_cut = XREG(P->y_u.xps.x); | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							|  |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s:%d (YENV=%p ENV=%p ASP=%p)\n", worker_id, | 
					
						
							|  |  |  |             LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, __FUNCTION__, | 
					
						
							|  |  |  |             __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(2 PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (Yap_only_has_signals(YAP_CDOVF_SIGNAL, YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     return 2; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (PP) | 
					
						
							|  |  |  |     UNLOCKPE(1, PP); | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |   PP = P->y_u.xps.p0; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   /* find something to fool S */ | 
					
						
							|  |  |  |   if (P->opc == Yap_opcode(_fcall)) { | 
					
						
							|  |  |  |     /* fill it up */ | 
					
						
							|  |  |  |     CACHE_Y_AS_ENV(YREG); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     ENV_YREG[E_CP] = (CELL)CP; | 
					
						
							|  |  |  |     ENV_YREG[E_E] = (CELL)ENV; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #ifdef DEPTH_LIMIT
 | 
					
						
							|  |  |  |     ENV_YREG[E_DEPTH] = DEPTH; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | #endif /* DEPTH_LIMIT */
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     ENDCACHE_Y_AS_ENV(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   P = NEXTOP(P, xps); | 
					
						
							|  |  |  |   return interrupt_handler_either(t_cut, PredRestoreRegs PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_either(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-30 07:25:34 +00:00
										 |  |  | #ifdef DEBUGX
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   // if (trace_interrupts)
 | 
					
						
							|  |  |  |   fprintf(stderr, "[%d] %s:%d:  (YENV=%p ENV=%p ASP=%p)\n", worker_id, | 
					
						
							|  |  |  |           __FUNCTION__, __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = check_alarm_fail_int(2 PASS_REGS)) >= 0) { | 
					
						
							| 
									
										
										
										
											2013-12-16 13:05:08 +00:00
										 |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (Yap_only_has_signal(YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     return 2; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (PP) | 
					
						
							|  |  |  |     UNLOCKPE(1, PP); | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |   PP = P->y_u.Osblp.p0; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   /* find something to fool S */ | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |   SET_ASP(YENV, P->y_u.Osbpp.s); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   if (ASP > (CELL *)PROTECT_FROZEN_B(B)) | 
					
						
							|  |  |  |     ASP = (CELL *)PROTECT_FROZEN_B(B); | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   if ((v = code_overflow(YENV PASS_REGS)) >= 0) { | 
					
						
							|  |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   // P = NEXTOP(P, Osblp);
 | 
					
						
							|  |  |  |   if ((v = stack_overflow( | 
					
						
							|  |  |  |            RepPredProp(Yap_GetPredPropByFunc(FunctorRestoreRegs1, 0)), YENV, | 
					
						
							|  |  |  |            NEXTOP(P, Osblp), 0 PASS_REGS)) >= 0) { | 
					
						
							|  |  |  |     // P = PREVOP(P, Osblp);
 | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-01-30 07:25:34 +00:00
										 |  |  |   // P = PREVOP(P, Osblp);
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   return interrupt_handler_either( | 
					
						
							|  |  |  |       MkIntTerm(0), | 
					
						
							|  |  |  |       RepPredProp(Yap_GetPredPropByFunc(FunctorRestoreRegs1, 0)) PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static int interrupt_dexecute(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   int v; | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   PredEntry *pe; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #ifdef DEBUG_INTERRUPTS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (trace_interrupts) | 
					
						
							|  |  |  |     fprintf(stderr, "[%d] %lu--%lu %s/%d (YENV=%p ENV=%p ASP=%p)\n", worker_id, | 
					
						
							|  |  |  |             LOCAL_FirstActiveSignal, LOCAL_LastActiveSignal, __FUNCTION__, | 
					
						
							|  |  |  |             __LINE__, YENV, ENV, ASP); | 
					
						
							| 
									
										
										
										
											2013-12-14 12:35:18 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (PP) | 
					
						
							|  |  |  |     UNLOCKPE(1, PP); | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |   PP = P->y_u.pp.p0; | 
					
						
							|  |  |  |   pe = P->y_u.pp.p; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((pe->PredFlags & (NoTracePredFlag | HiddenPredFlag)) && | 
					
						
							|  |  |  |       Yap_only_has_signal(YAP_CREEP_SIGNAL)) { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     return 2; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* set S for next instructions */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   ASP = YENV + E_CB; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   if (ASP > (CELL *)PROTECT_FROZEN_B(B)) | 
					
						
							|  |  |  |     ASP = (CELL *)PROTECT_FROZEN_B(B); | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   if ((v = code_overflow(YENV PASS_REGS)) >= 0) { | 
					
						
							|  |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if ((v = stack_overflow(P->y_u.pp.p, (CELL *)YENV[E_E], (yamop *)YENV[E_CP], | 
					
						
							|  |  |  |                           P->y_u.pp.p->ArityOfPE PASS_REGS)) >= 0) { | 
					
						
							|  |  |  |     return v; | 
					
						
							| 
									
										
										
										
											2014-04-23 22:41:12 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   /* first, deallocate */ | 
					
						
							|  |  |  |   CP = (yamop *)YENV[E_CP]; | 
					
						
							|  |  |  |   ENV = YENV = (CELL *)YENV[E_E]; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #ifdef DEPTH_LIMIT
 | 
					
						
							|  |  |  |   YENV[E_DEPTH] = DEPTH; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | #endif /* DEPTH_LIMIT */
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #ifdef FROZEN_STACKS
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |     choiceptr top_b = PROTECT_FROZEN_B(B); | 
					
						
							|  |  |  | #ifdef YAPOR_SBA
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (YENV > (CELL *)top_b || YENV < HR) | 
					
						
							|  |  |  |       YENV = (CELL *)top_b; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (YENV > (CELL *)top_b) | 
					
						
							|  |  |  |       YENV = (CELL *)top_b; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #endif /* YAPOR_SBA */
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     else | 
					
						
							|  |  |  |       YENV = (CELL *)((CELL)YENV + ENV_Size(CPREG)); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (YENV > (CELL *)B) { | 
					
						
							|  |  |  |     YENV = (CELL *)B; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     YENV = (CELL *)((CELL)YENV + ENV_Size(CPREG)); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | #endif /* FROZEN_STACKS */
 | 
					
						
							|  |  |  |   /* setup GB */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   YENV[E_CB] = (CELL)B; | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* and now CREEP */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   return interrupt_handler(pe PASS_REGS); | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static void undef_goal(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |   PredEntry *pe = PredFromDefCode(P); | 
					
						
							| 
									
										
										
										
											2016-05-12 11:42:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |   BEGD(d0); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | /* avoid trouble with undefined dynamic procedures */ | 
					
						
							|  |  |  | /* I assume they were not locked beforehand */ | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							|  |  |  |   if (!PP) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     PELOCK(19, pe); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     PP = pe; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (pe->PredFlags & (DynamicPredFlag | LogUpdatePredFlag | MultiFileFlag)) { | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     UNLOCKPE(19, PP); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     PP = NULL; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     P = FAILCODE; | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   UNLOCKPE(19, PP); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |   PP = NULL; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   d0 = pe->ArityOfPE; | 
					
						
							|  |  |  |   if (d0 == 0) { | 
					
						
							|  |  |  |     HR[1] = MkAtomTerm((Atom)(pe->FunctorOfPred)); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     HR[d0 + 2] = AbsAppl(HR); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     *HR = (CELL)pe->FunctorOfPred; | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     HR++; | 
					
						
							|  |  |  |     BEGP(pt1); | 
					
						
							|  |  |  |     pt1 = XREGS + 1; | 
					
						
							|  |  |  |     for (; d0 > 0; --d0) { | 
					
						
							|  |  |  |       BEGD(d1); | 
					
						
							|  |  |  |       BEGP(pt0); | 
					
						
							|  |  |  |       pt0 = pt1++; | 
					
						
							|  |  |  |       d1 = *pt0; | 
					
						
							|  |  |  |       deref_head(d1, undef_unk); | 
					
						
							|  |  |  |     undef_nonvar: | 
					
						
							|  |  |  |       /* just copy it to the heap */ | 
					
						
							|  |  |  |       *HR++ = d1; | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       derefa_body(d1, pt0, undef_unk, undef_nonvar); | 
					
						
							|  |  |  |       if (pt0 <= HR) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         /* variable is safe */ | 
					
						
							|  |  |  |         *HR++ = (CELL)pt0; | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         /* bind it, in case it is a local variable */ | 
					
						
							|  |  |  |         d1 = Unsigned(HR); | 
					
						
							|  |  |  |         RESET_VARIABLE(HR); | 
					
						
							|  |  |  |         HR += 1; | 
					
						
							|  |  |  |         Bind_Local(pt0, d1); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |       ENDP(pt0); | 
					
						
							|  |  |  |       ENDD(d1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ENDP(pt1); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   ENDD(d0); | 
					
						
							|  |  |  |   HR[0] = Yap_Module_Name(pe); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   ARG1 = (Term)AbsPair(HR); | 
					
						
							| 
									
										
										
										
											2015-06-17 23:53:13 +01:00
										 |  |  |   ARG2 = Yap_getUnknownModule(Yap_GetModuleEntry(HR[0])); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |   HR += 2; | 
					
						
							|  |  |  | #ifdef LOW_LEVEL_TRACER
 | 
					
						
							|  |  |  |   if (Yap_do_low_level_trace) | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     low_level_trace(enter_pred, UndefCode, XREGS + 1); | 
					
						
							|  |  |  | #endif /* LOW_LEVEL_TRACE */
 | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |   P = UndefCode->CodeOfPred; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | static void spy_goal(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |   PredEntry *pe = PredFromDefCode(P); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							|  |  |  |   if (!PP) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     PELOCK(14, pe); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     PP = pe; | 
					
						
							| 
									
										
										
										
											2015-01-18 03:00:19 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   BEGD(d0); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (!(pe->PredFlags & IndexedPredFlag) && pe->cs.p_code.NOfClauses > 1) { | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     /* update ASP before calling IPred */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     SET_ASP(YREG, E_CB * sizeof(CELL)); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     Yap_IPred(pe, 0, CP); | 
					
						
							|  |  |  |     /* IPred can generate errors, it thus must get rid of the lock itself */ | 
					
						
							|  |  |  |     if (P == FAILCODE) { | 
					
						
							|  |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       if (PP && !(PP->PredFlags & LogUpdatePredFlag)) { | 
					
						
							|  |  |  |         UNLOCKPE(20, pe); | 
					
						
							|  |  |  |         PP = NULL; | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* first check if we need to increase the counter */ | 
					
						
							|  |  |  |   if ((pe->PredFlags & CountPredFlag)) { | 
					
						
							| 
									
										
										
										
											2015-04-21 16:09:43 -06:00
										 |  |  |     LOCK(pe->StatisticsForPred->lock); | 
					
						
							|  |  |  |     pe->StatisticsForPred->NOfEntries++; | 
					
						
							|  |  |  |     UNLOCK(pe->StatisticsForPred->lock); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     LOCAL_ReductionsCounter--; | 
					
						
							|  |  |  |     if (LOCAL_ReductionsCounter == 0 && LOCAL_ReductionsCounterOn) { | 
					
						
							|  |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							|  |  |  |       if (PP) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         UNLOCKPE(20, pe); | 
					
						
							|  |  |  |         PP = NULL; | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       Yap_NilError(CALL_COUNTER_UNDERFLOW_EVENT, ""); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     LOCAL_PredEntriesCounter--; | 
					
						
							|  |  |  |     if (LOCAL_PredEntriesCounter == 0 && LOCAL_PredEntriesCounterOn) { | 
					
						
							|  |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							|  |  |  |       if (PP) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         UNLOCKPE(21, pe); | 
					
						
							|  |  |  |         PP = NULL; | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       Yap_NilError(PRED_ENTRY_COUNTER_UNDERFLOW_EVENT, ""); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if ((pe->PredFlags & (CountPredFlag | ProfiledPredFlag | SpiedPredFlag)) == | 
					
						
							|  |  |  |         CountPredFlag) { | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							|  |  |  |       if (PP) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         UNLOCKPE(22, pe); | 
					
						
							|  |  |  |         PP = NULL; | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       P = pe->cs.p_code.TrueCodeOfPred; | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* standard profiler */ | 
					
						
							|  |  |  |   if ((pe->PredFlags & ProfiledPredFlag)) { | 
					
						
							| 
									
										
										
										
											2016-05-13 11:41:19 +01:00
										 |  |  |     if (!pe->StatisticsForPred) | 
					
						
							|  |  |  |       Yap_initProfiler(pe); | 
					
						
							| 
									
										
										
										
											2015-04-21 16:09:43 -06:00
										 |  |  |     LOCK(pe->StatisticsForPred->lock); | 
					
						
							|  |  |  |     pe->StatisticsForPred->NOfEntries++; | 
					
						
							|  |  |  |     UNLOCK(pe->StatisticsForPred->lock); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     if (!(pe->PredFlags & SpiedPredFlag)) { | 
					
						
							|  |  |  |       P = pe->cs.p_code.TrueCodeOfPred; | 
					
						
							|  |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							|  |  |  |       if (PP) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         UNLOCKPE(23, pe); | 
					
						
							|  |  |  |         PP = NULL; | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #if defined(YAPOR) || defined(THREADS)
 | 
					
						
							|  |  |  |   if (PP) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     UNLOCKPE(25, pe); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     PP = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   d0 = pe->ArityOfPE; | 
					
						
							|  |  |  |   /* save S for ModuleName */ | 
					
						
							|  |  |  |   if (d0 == 0) { | 
					
						
							|  |  |  |     HR[1] = MkAtomTerm((Atom)(pe->FunctorOfPred)); | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     *HR = (CELL)pe->FunctorOfPred; | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     HR[d0 + 2] = AbsAppl(HR); | 
					
						
							|  |  |  |     HR++; | 
					
						
							|  |  |  |     BEGP(pt1); | 
					
						
							|  |  |  |     pt1 = XREGS + 1; | 
					
						
							|  |  |  |     for (; d0 > 0; --d0) { | 
					
						
							|  |  |  |       BEGD(d1); | 
					
						
							|  |  |  |       BEGP(pt0); | 
					
						
							|  |  |  |       pt0 = pt1++; | 
					
						
							|  |  |  |       d1 = *pt0; | 
					
						
							|  |  |  |       deref_head(d1, dospy_unk); | 
					
						
							|  |  |  |     dospy_nonvar: | 
					
						
							|  |  |  |       /* just copy it to the heap */ | 
					
						
							|  |  |  |       *HR++ = d1; | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       derefa_body(d1, pt0, dospy_unk, dospy_nonvar); | 
					
						
							|  |  |  |       if (pt0 <= HR) { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         /* variable is safe */ | 
					
						
							|  |  |  |         *HR++ = (CELL)pt0; | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |         /* bind it, in case it is a local variable */ | 
					
						
							|  |  |  |         d1 = Unsigned(HR); | 
					
						
							|  |  |  |         RESET_VARIABLE(HR); | 
					
						
							|  |  |  |         HR += 1; | 
					
						
							|  |  |  |         Bind_Local(pt0, d1); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |       ENDP(pt0); | 
					
						
							|  |  |  |       ENDD(d1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ENDP(pt1); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   ENDD(d0); | 
					
						
							|  |  |  |   HR[0] = Yap_Module_Name(pe); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   ARG1 = (Term)AbsPair(HR); | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |   HR += 2; | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     PredEntry *pt0; | 
					
						
							| 
									
										
										
										
											2014-11-14 16:51:42 +00:00
										 |  |  | #if THREADS
 | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     LOCK(GLOBAL_ThreadHandlesLock); | 
					
						
							| 
									
										
										
										
											2014-11-14 16:51:42 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     pt0 = SpyCode; | 
					
						
							|  |  |  |     P_before_spy = P; | 
					
						
							|  |  |  |     P = pt0->CodeOfPred; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | /* for profiler */ | 
					
						
							| 
									
										
										
										
											2014-11-14 16:51:42 +00:00
										 |  |  | #if THREADS
 | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |     UNLOCK(GLOBAL_ThreadHandlesLock); | 
					
						
							| 
									
										
										
										
											2014-11-14 16:51:42 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  | #ifdef LOW_LEVEL_TRACER
 | 
					
						
							|  |  |  |     if (Yap_do_low_level_trace) | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       low_level_trace(enter_pred, pt0, XREGS + 1); | 
					
						
							|  |  |  | #endif /* LOW_LEVEL_TRACE */
 | 
					
						
							| 
									
										
										
										
											2014-10-22 10:10:43 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | Int Yap_absmi(int inp) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if BP_FREE
 | 
					
						
							|  |  |  |   /* some function might be using bp for an internal variable, it is the
 | 
					
						
							|  |  |  |      callee's responsability to save it */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   yamop *PCBACKUP = P1REG; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef LONG_LIVED_REGISTERS
 | 
					
						
							|  |  |  |   register CELL d0, d1; | 
					
						
							|  |  |  |   register CELL *pt0, *pt1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* LONG_LIVED_REGISTERS */
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef SHADOW_P
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   register yamop *PREG = P; | 
					
						
							|  |  |  | #endif /* SHADOW_P */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-26 14:37:08 +00:00
										 |  |  | #ifdef SHADOW_CP
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   register yamop *CPREG = CP; | 
					
						
							|  |  |  | #endif /* SHADOW_CP */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-26 14:37:08 +00:00
										 |  |  | #ifdef SHADOW_HB
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   register CELL *HBREG = HB; | 
					
						
							|  |  |  | #endif /* SHADOW_HB */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-26 14:37:08 +00:00
										 |  |  | #ifdef SHADOW_Y
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   register CELL *YREG = Yap_REGS.YENV_; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* SHADOW_Y */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-26 14:37:08 +00:00
										 |  |  | #ifdef SHADOW_S
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   register CELL *SREG = Yap_REGS.S_; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  | #define SREG S
 | 
					
						
							|  |  |  | #endif /* SHADOW_S */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | /* The indexing register so that we will not destroy ARG1 without
 | 
					
						
							|  |  |  |  * reason */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #define I_R (XREGS[0])
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-18 03:00:19 +00:00
										 |  |  | #if YAP_JIT
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   Yap_ExpEnvP = &Yap_ExpEnv; | 
					
						
							|  |  |  |   static void *control_labels[] = { | 
					
						
							|  |  |  |       &&fail,        &&NoStackCut,      &&NoStackCommitY, | 
					
						
							|  |  |  |       &&NoStackCutT, &&NoStackEither,   &&NoStackExecute, | 
					
						
							|  |  |  |       &&NoStackCall, &&NoStackDExecute, &&NoStackDeallocate, | 
					
						
							|  |  |  |       &¬railleft, &&NoStackFail,     &&NoStackCommitX}; | 
					
						
							| 
									
										
										
										
											2015-01-18 03:00:19 +00:00
										 |  |  |   curtrace = NULL; | 
					
						
							|  |  |  |   curpreg = NULL; | 
					
						
							|  |  |  |   globalcurblock = NULL; | 
					
						
							|  |  |  |   ineedredefinedest = 0; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   NativeArea = (NativeContext *)malloc(sizeof(NativeContext)); | 
					
						
							| 
									
										
										
										
											2015-01-18 03:00:19 +00:00
										 |  |  |   NativeArea->area.p = NULL; | 
					
						
							|  |  |  |   NativeArea->area.ok = NULL; | 
					
						
							|  |  |  |   NativeArea->area.pc = NULL; | 
					
						
							|  |  |  | #if YAP_STAT_PREDS
 | 
					
						
							|  |  |  |   NativeArea->area.nrecomp = NULL; | 
					
						
							|  |  |  |   NativeArea->area.compilation_time = NULL; | 
					
						
							|  |  |  |   NativeArea->area.native_size_bytes = NULL; | 
					
						
							|  |  |  |   NativeArea->area.trace_size_bytes = NULL; | 
					
						
							|  |  |  |   NativeArea->success = NULL; | 
					
						
							|  |  |  |   NativeArea->runs = NULL; | 
					
						
							|  |  |  |   NativeArea->t_runs = NULL; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   NativeArea->n = 0; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   IntermediatecodeArea = | 
					
						
							|  |  |  |       (IntermediatecodeContext *)malloc(sizeof(IntermediatecodeContext)); | 
					
						
							| 
									
										
										
										
											2015-01-18 03:00:19 +00:00
										 |  |  |   IntermediatecodeArea->area.t = NULL; | 
					
						
							|  |  |  |   IntermediatecodeArea->area.ok = NULL; | 
					
						
							|  |  |  |   IntermediatecodeArea->area.isactive = NULL; | 
					
						
							|  |  |  |   IntermediatecodeArea->area.lastblock = NULL; | 
					
						
							|  |  |  | #if YAP_STAT_PREDS
 | 
					
						
							|  |  |  |   IntermediatecodeArea->area.profiling_time = NULL; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   IntermediatecodeArea->n = 0; | 
					
						
							|  |  |  |   nnexec = 0; | 
					
						
							|  |  |  |   l = 0; | 
					
						
							|  |  |  | #endif /* YAP_JIT */
 | 
					
						
							| 
									
										
										
										
											2015-04-13 13:14:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if USE_THREADED_CODE
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   /************************************************************************/ | 
					
						
							|  |  |  |   /*     Abstract Machine Instruction Address Table                       */ | 
					
						
							|  |  |  |   /*  This must be declared inside the function. We use the asm directive */ | 
					
						
							|  |  |  |   /* to make it available outside this function                           */ | 
					
						
							|  |  |  |   /************************************************************************/ | 
					
						
							|  |  |  |   static void *OpAddress[] = { | 
					
						
							|  |  |  | #define OPCODE(OP, TYPE) &&_##OP
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #include "YapOpcodes.h"
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | #undef OPCODE
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-18 03:00:19 +00:00
										 |  |  | #if YAP_JIT
 | 
					
						
							| 
									
										
										
										
											2015-02-06 18:11:52 +00:00
										 |  |  |   ExpEnv.config_struc.TOTAL_OF_OPCODES = | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       sizeof(OpAddress) / (2 * sizeof(void *)); | 
					
						
							| 
									
										
										
										
											2015-01-18 03:00:19 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-04-13 13:14:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* USE_THREADED_CODE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | /*static void* (*nat_glist_valx)(yamop**,yamop**,CELL**,void**,int*);
 | 
					
						
							| 
									
										
										
										
											2015-04-13 13:14:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (nat_glist_valx == NULL) { | 
					
						
							|  |  |  |   nat_glist_valx = | 
					
						
							|  |  |  |   (void*(*)(yamop**,yamop**,CELL**,void**,int*))call_JIT_Compiler(J, | 
					
						
							|  |  |  |   _glist_valx); | 
					
						
							|  |  |  |   }*/ | 
					
						
							| 
									
										
										
										
											2015-04-13 13:14:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-26 14:37:08 +00:00
										 |  |  | #ifdef SHADOW_REGS
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* work with a local pointer to the registers */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   register REGSTORE *regp = &Yap_REGS; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif /* SHADOW_REGS */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if PUSH_REGS
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   /* useful on a X86 with -fomit-frame-pointer optimisation */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* The idea is to push REGS onto the X86 stack frame */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* first allocate local space */ | 
					
						
							|  |  |  |   REGSTORE absmi_regs; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   REGSTORE *old_regs = Yap_regp; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif /* PUSH_REGS */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   CELL OLD_B = B; | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  |   extern PredEntry *bpEntry; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   if (inp == -9000) { | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  | #if PUSH_REGS
 | 
					
						
							|  |  |  |     old_regs = &Yap_REGS; | 
					
						
							|  |  |  |     init_absmi_regs(&absmi_regs); | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | #if THREADS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     regcache = Yap_regp LOCAL_PL_local_data_p->reg_cache = regcache; | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  |     Yap_regp = &absmi_regs; | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-04-12 17:14:58 +00:00
										 |  |  |     CACHE_A1(); | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     PREG = bpEntry->CodeOfPred; | 
					
						
							|  |  |  |     JMPNext(); /* go execute instruction at PREG */ | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if USE_THREADED_CODE
 | 
					
						
							|  |  |  |   /* absmadr */ | 
					
						
							|  |  |  |   if (inp > 0) { | 
					
						
							| 
									
										
										
										
											2015-02-07 01:08:32 +00:00
										 |  |  |     Yap_ABSMI_OPCODES = OpAddress; | 
					
						
							|  |  |  | #if YAP_JIT
 | 
					
						
							|  |  |  |     Yap_ABSMI_ControlLabels = control_labels; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if BP_FREE
 | 
					
						
							|  |  |  |     P1REG = PCBACKUP; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     return (0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | #endif /* USE_THREADED_CODE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if PUSH_REGS
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   old_regs = &Yap_REGS; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 15:04:12 +00:00
										 |  |  |   /* done, let us now initialize this space */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   init_absmi_regs(&absmi_regs); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | /* the registers are all set up, let's swap */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #ifdef THREADS
 | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  |   pthread_setspecific(Yap_yaamregs_key, (const void *)&absmi_regs); | 
					
						
							| 
									
										
										
										
											2011-05-09 19:55:06 +01:00
										 |  |  |   LOCAL_ThreadHandle.current_yaam_regs = &absmi_regs; | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   regcache = &absmi_regs; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | // LOCAL_PL_local_data_p->reg_cache = regcache;
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   Yap_regp = &absmi_regs; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | #undef Yap_REGS
 | 
					
						
							|  |  |  | #define Yap_REGS absmi_regs
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif /* PUSH_REGS */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-26 14:37:08 +00:00
										 |  |  | #ifdef SHADOW_REGS
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* use regp as a copy of REGS */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   regp = &Yap_REGS; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef REGS
 | 
					
						
							|  |  |  | #undef REGS
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #define REGS (*regp)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* SHADOW_REGS */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   setregs(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   CACHE_A1(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | reset_absmi: | 
					
						
							| 
									
										
										
										
											2014-05-25 20:43:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   SP = SP0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if USE_THREADED_CODE
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   //___androidlog_print(ANDROID_LOG_INFO, "YAP ", "%s",
 | 
					
						
							|  |  |  |   // Yap_op_names[Yap_op_from_opcode(PREG->opc)]);
 | 
					
						
							| 
									
										
										
										
											2015-04-13 13:14:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |   JMPNext(); /* go execute instruction at P          */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   /* when we start we are not in write mode */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     op_numbers opcode = _Ystop; | 
					
						
							|  |  |  |     op_numbers old_op; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | #ifdef DEBUG_XX
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     unsigned long ops_done; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     goto nextop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   nextop_write: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     old_op = opcode; | 
					
						
							| 
									
										
										
										
											2014-05-30 01:06:09 +01:00
										 |  |  |     opcode = PREG->y_u.o.opcw; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     goto op_switch; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   nextop: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     old_op = opcode; | 
					
						
							|  |  |  |     opcode = PREG->opc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   op_switch: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef ANALYST
 | 
					
						
							| 
									
										
										
										
											2011-05-25 16:40:36 +01:00
										 |  |  |     GLOBAL_opcount[opcode]++; | 
					
						
							|  |  |  |     GLOBAL_2opcount[old_op][opcode]++; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef DEBUG_XX
 | 
					
						
							|  |  |  |     ops_done++; | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | /*    if (B->cp_b > 0x103fff90)
 | 
					
						
							|  |  |  |   fprintf(stderr,"(%ld) doing %s, done %s, B is %p, HB is %p, H is %p\n", | 
					
						
							|  |  |  |   ops_done,Yap_op_names[opcode],Yap_op_names[old_op],B,B->cp_h,HR);*/ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #endif /* ANALYST */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (opcode) { | 
					
						
							|  |  |  | #endif /* USE_THREADED_CODE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !OS_HANDLES_TR_OVERFLOW
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | notrailleft: | 
					
						
							|  |  |  |   /* if we are within indexing code, the system may have to
 | 
					
						
							|  |  |  |    * update a S */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     CELL cut_b; | 
					
						
							| 
									
										
										
										
											2007-04-10 22:13:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-26 14:37:08 +00:00
										 |  |  | #ifdef SHADOW_S
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     S = SREG; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     /* YREG was pointing to where we were going to build the
 | 
					
						
							|  |  |  |      * next choice-point. The stack shifter will need to know this | 
					
						
							|  |  |  |      * to move the local stack */ | 
					
						
							|  |  |  |     SET_ASP(YREG, E_CB * sizeof(CELL)); | 
					
						
							|  |  |  |     cut_b = LCL0 - (CELL *)(ASP[E_CB]); | 
					
						
							|  |  |  |     saveregs(); | 
					
						
							|  |  |  |     if (!Yap_growtrail(0, false)) { | 
					
						
							|  |  |  |       Yap_NilError(RESOURCE_ERROR_TRAIL, | 
					
						
							|  |  |  |                    "YAP failed to reserve %ld bytes in growtrail", | 
					
						
							|  |  |  |                    sizeof(CELL) * K16); | 
					
						
							|  |  |  |       setregs(); | 
					
						
							|  |  |  |       FAIL(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     setregs(); | 
					
						
							| 
									
										
										
										
											2009-04-18 15:22:51 -05:00
										 |  |  | #ifdef SHADOW_S
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     SREG = S; | 
					
						
							| 
									
										
										
										
											2009-04-18 15:22:51 -05:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |     if (SREG == ASP) { | 
					
						
							|  |  |  |       SREG[E_CB] = (CELL)(LCL0 - cut_b); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   goto reset_absmi; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif /* OS_HANDLES_TR_OVERFLOW */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-18 03:00:19 +00:00
										 |  |  | // move instructions to separate file
 | 
					
						
							|  |  |  | // so that they are easier to analyse.
 | 
					
						
							|  |  |  | #include "absmi_insts.h"
 | 
					
						
							| 
									
										
										
										
											2001-10-30 16:42:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if !USE_THREADED_CODE
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | default: | 
					
						
							|  |  |  |   saveregs(); | 
					
						
							|  |  |  |   Yap_Error(SYSTEM_ERROR_INTERNAL, MkIntegerTerm(opcode), | 
					
						
							|  |  |  |             "trying to execute invalid YAAM instruction %d", opcode); | 
					
						
							|  |  |  |   setregs(); | 
					
						
							|  |  |  |   FAIL(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if PUSH_REGS
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       restore_absmi_regs(old_regs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #if BP_FREE
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       P1REG = PCBACKUP; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-12-12 14:24:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  |       return (0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-29 19:04:46 +00:00
										 |  |  | /* dummy function that is needed for profiler */ | 
					
						
							| 
									
										
										
										
											2016-03-29 01:55:12 +01:00
										 |  |  | int Yap_absmiEND(void) { return 1; } |