| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | /*************************************************************************
 | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *	 YAP Prolog 							 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *	Yap Prolog was developed at NCCUP - Universidade do Porto	 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | ************************************************************************** | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * File:		compiler.c						 * | 
					
						
							|  |  |  | * comments:	Clause compiler						 * | 
					
						
							|  |  |  | *									 * | 
					
						
							| 
									
										
										
										
											2008-08-06 17:32:22 +00:00
										 |  |  | * Last rev:     $Date: 2008-08-06 17:32:18 $,$Author: vsc $						 * | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  | * $Log: not supported by cvs2svn $ | 
					
						
							| 
									
										
										
										
											2008-08-06 17:32:22 +00:00
										 |  |  | * Revision 1.88  2008/03/13 14:37:58  vsc | 
					
						
							|  |  |  | * update chr | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2008-03-13 14:38:02 +00:00
										 |  |  | * Revision 1.87  2007/12/18 17:46:58  vsc | 
					
						
							|  |  |  | * purge_clauses does not need to do anything if there are no clauses | 
					
						
							|  |  |  | * fix gprof bugs. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2007-12-18 17:46:58 +00:00
										 |  |  | * Revision 1.86  2007/11/26 23:43:08  vsc | 
					
						
							|  |  |  | * fixes to support threads and assert correctly, even if inefficiently. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | * Revision 1.85  2007/11/06 17:02:11  vsc | 
					
						
							|  |  |  | * compile ground terms away. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | * Revision 1.84  2007/03/27 13:48:51  vsc | 
					
						
							|  |  |  | * fix number of overflows (comments by Bart Demoen). | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2007-03-27 13:48:51 +00:00
										 |  |  | * Revision 1.83  2007/03/26 15:18:43  vsc | 
					
						
							|  |  |  | * debugging and clause/3 over tabled predicates would kill YAP. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2007-03-26 15:18:43 +00:00
										 |  |  | * Revision 1.82  2006/11/06 18:35:03  vsc | 
					
						
							|  |  |  | * 1estranha | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-11-06 18:35:05 +00:00
										 |  |  | * Revision 1.81  2006/10/11 15:08:03  vsc | 
					
						
							|  |  |  | * fix bb entries | 
					
						
							|  |  |  | * comment development code for timestamp overflow. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-10-11 15:08:03 +00:00
										 |  |  | * Revision 1.80  2006/09/20 20:03:51  vsc | 
					
						
							|  |  |  | * improve indexing on floats | 
					
						
							|  |  |  | * fix sending large lists to DB | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-09-20 20:03:51 +00:00
										 |  |  | * Revision 1.79  2006/08/01 13:14:17  vsc | 
					
						
							|  |  |  | * fix compilation of | | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-08-01 13:14:17 +00:00
										 |  |  | * Revision 1.78  2006/07/27 19:04:56  vsc | 
					
						
							|  |  |  | * fix nasty overflows in and add some very preliminary support for very large | 
					
						
							|  |  |  | * clauses with lots | 
					
						
							|  |  |  | * of disjuncts (eg, query packs). | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-07-27 19:04:56 +00:00
										 |  |  | * Revision 1.77  2006/05/19 14:31:31  vsc | 
					
						
							|  |  |  | * get rid of IntArrays and FloatArray code. | 
					
						
							|  |  |  | * include holes when calculating memory usage. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-05-19 14:31:32 +00:00
										 |  |  | * Revision 1.76  2006/05/19 13:48:11  vsc | 
					
						
							|  |  |  | * help to make Yap work with dynamic libs | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-05-19 13:48:11 +00:00
										 |  |  | * Revision 1.75  2006/05/16 18:37:30  vsc | 
					
						
							|  |  |  | * WIN32 fixes | 
					
						
							|  |  |  | * compiler bug fixes | 
					
						
							|  |  |  | * extend interface | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-05-16 18:37:31 +00:00
										 |  |  | * Revision 1.74  2006/04/13 02:04:24  vsc | 
					
						
							|  |  |  | * fix debugging typo | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-04-13 02:04:24 +00:00
										 |  |  | * Revision 1.73  2006/04/12 20:08:51  vsc | 
					
						
							|  |  |  | * make it sure that making vars safe does not propagate across branches of disjunctions. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-04-12 20:08:51 +00:00
										 |  |  | * Revision 1.72  2006/04/05 00:16:54  vsc | 
					
						
							|  |  |  | * Lots of fixes (check logfile for details | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-04-05 00:16:55 +00:00
										 |  |  | * Revision 1.71  2006/03/24 17:13:41  rslopes | 
					
						
							|  |  |  | * New update to BEAM engine. | 
					
						
							|  |  |  | * BEAM now uses YAP Indexing (JITI) | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  | * Revision 1.70  2005/12/17 03:25:39  vsc | 
					
						
							|  |  |  | * major changes to support online event-based profiling | 
					
						
							|  |  |  | * improve error discovery and restart on scanner. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-12-17 03:25:39 +00:00
										 |  |  | * Revision 1.69  2005/09/08 22:06:44  rslopes | 
					
						
							|  |  |  | * BEAM for YAP update... | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | * Revision 1.68  2005/07/06 15:10:03  vsc | 
					
						
							|  |  |  | * improvements to compiler: merged instructions and fixes for -> | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | * Revision 1.67  2005/05/25 21:43:32  vsc | 
					
						
							|  |  |  | * fix compiler bug in 1 << X, found by Nuno Fonseca. | 
					
						
							|  |  |  | * compiler internal errors get their own message. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-05-25 21:43:33 +00:00
										 |  |  | * Revision 1.66  2005/05/12 03:36:32  vsc | 
					
						
							|  |  |  | * debugger was making predicates meta instead of testing | 
					
						
							|  |  |  | * fix handling of dbrefs in facts and in subarguments. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-05-12 03:36:33 +00:00
										 |  |  | * Revision 1.65  2005/04/10 04:01:10  vsc | 
					
						
							|  |  |  | * bug fixes, I hope! | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-04-10 04:01:15 +00:00
										 |  |  | * Revision 1.64  2005/03/13 06:26:10  vsc | 
					
						
							|  |  |  | * fix excessive pruning in meta-calls | 
					
						
							|  |  |  | * fix Term->int breakage in compiler | 
					
						
							|  |  |  | * improve JPL (at least it does something now for amd64). | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-03-13 06:26:13 +00:00
										 |  |  | * Revision 1.63  2005/03/04 20:30:11  ricroc | 
					
						
							|  |  |  | * bug fixes for YapTab support | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-03-04 20:30:14 +00:00
										 |  |  | * Revision 1.62  2005/02/21 16:49:39  vsc | 
					
						
							|  |  |  | * amd64 fixes | 
					
						
							|  |  |  | * library fixes | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-02-21 16:50:21 +00:00
										 |  |  | * Revision 1.61  2005/01/28 23:14:35  vsc | 
					
						
							|  |  |  | * move to Yap-4.5.7 | 
					
						
							|  |  |  | * Fix clause size | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-01-28 23:14:41 +00:00
										 |  |  | * Revision 1.60  2005/01/14 20:55:16  vsc | 
					
						
							|  |  |  | * improve register liveness calculations. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  | * Revision 1.59  2005/01/04 02:50:21  vsc | 
					
						
							|  |  |  | * - allow MegaClauses with blobs | 
					
						
							|  |  |  | * - change Diffs to be thread specific | 
					
						
							|  |  |  | * - include Christian's updates | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-01-04 02:50:23 +00:00
										 |  |  | * Revision 1.58  2005/01/03 17:06:03  vsc | 
					
						
							|  |  |  | * fix discontiguous stack overflows in parser | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2005-01-03 17:06:05 +00:00
										 |  |  | * Revision 1.57  2004/12/20 21:44:57  vsc | 
					
						
							|  |  |  | * more fixes to CLPBN | 
					
						
							|  |  |  | * fix some Yap overflows. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2004-12-20 21:44:58 +00:00
										 |  |  | * Revision 1.56  2004/12/16 05:57:32  vsc | 
					
						
							|  |  |  | * fix overflows | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2004-12-16 05:57:32 +00:00
										 |  |  | * Revision 1.55  2004/12/05 05:01:23  vsc | 
					
						
							|  |  |  | * try to reduce overheads when running with goal expansion enabled. | 
					
						
							|  |  |  | * CLPBN fixes | 
					
						
							|  |  |  | * Handle overflows when allocating big clauses properly. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2004-12-05 05:01:45 +00:00
										 |  |  | * Revision 1.54  2004/11/19 22:08:41  vsc | 
					
						
							|  |  |  | * replace SYSTEM_ERROR by out OUT_OF_WHATEVER_ERROR whenever appropriate. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2004-11-19 22:08:43 +00:00
										 |  |  | * Revision 1.53  2004/09/03 03:11:08  vsc | 
					
						
							|  |  |  | * memory management fixes | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2004-09-03 03:11:09 +00:00
										 |  |  | * Revision 1.52  2004/07/15 17:20:23  vsc | 
					
						
							|  |  |  | * fix error message | 
					
						
							|  |  |  | * change makefile and configure for clpbn | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2004-07-15 17:20:23 +00:00
										 |  |  | * Revision 1.51  2004/06/29 19:04:41  vsc | 
					
						
							|  |  |  | * fix multithreaded version | 
					
						
							|  |  |  | * include new version of Ricardo's profiler | 
					
						
							|  |  |  | * new predicat atomic_concat | 
					
						
							|  |  |  | * allow multithreaded-debugging | 
					
						
							|  |  |  | * small fixes | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2004-06-29 19:04:46 +00:00
										 |  |  | * Revision 1.50  2004/04/22 20:07:04  vsc | 
					
						
							|  |  |  | * more fixes for USE_SYSTEM_MEMORY | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2004-04-22 20:07:07 +00:00
										 |  |  | * Revision 1.49  2004/03/10 16:27:39  vsc | 
					
						
							|  |  |  | * skip compilation steps for ground facts. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  | * Revision 1.48  2004/03/08 19:31:01  vsc | 
					
						
							|  |  |  | * move to 4.5.3 | 
					
						
							|  |  |  | *									 * | 
					
						
							| 
									
										
										
										
											2004-03-08 19:31:01 +00:00
										 |  |  | *									 * | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | *************************************************************************/ | 
					
						
							|  |  |  | #ifdef SCCS
 | 
					
						
							|  |  |  | static char SccsId[] = "%W% %G%"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif	/* SCCS */
 | 
					
						
							|  |  |  | #include "Yap.h"
 | 
					
						
							|  |  |  | #include "compile.h"
 | 
					
						
							|  |  |  | #include "clause.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-08 00:53:38 +01:00
										 |  |  | #include "alloc.h"
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #include "yapio.h"
 | 
					
						
							|  |  |  | #if HAVE_STRING_H
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  | extern int EAM; | 
					
						
							|  |  |  | //extern PInstr *CodeStart, *ppc, *ppc1, *BodyStart, *ppc_body;
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | typedef struct branch_descriptor { | 
					
						
							|  |  |  |   int    id;                /* the branch id */ | 
					
						
							|  |  |  |   Term   cm;               /* if a banch is associated with a commit */ | 
					
						
							|  |  |  | } branch; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | typedef struct compiler_struct_struct { | 
					
						
							|  |  |  |   branch parent_branches[256]; | 
					
						
							|  |  |  |   branch *branch_pointer; | 
					
						
							|  |  |  |   PInstr *BodyStart; | 
					
						
							|  |  |  |   Ventry *vtable; | 
					
						
							|  |  |  |   CExpEntry *common_exps; | 
					
						
							| 
									
										
										
										
											2005-01-28 23:14:41 +00:00
										 |  |  |   int is_a_fact; | 
					
						
							| 
									
										
										
										
											2005-05-12 03:36:33 +00:00
										 |  |  |   int hasdbrefs; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   int n_common_exps; | 
					
						
							|  |  |  |   int goalno; | 
					
						
							|  |  |  |   int onlast; | 
					
						
							|  |  |  |   int onhead; | 
					
						
							|  |  |  |   int onbranch; | 
					
						
							|  |  |  |   int curbranch; | 
					
						
							| 
									
										
										
										
											2010-08-02 13:04:30 +01:00
										 |  |  |   Int space_used; | 
					
						
							|  |  |  |   PInstr *space_op; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   Prop current_p0; | 
					
						
							|  |  |  | #ifdef TABLING_INNER_CUTS
 | 
					
						
							|  |  |  |   PInstr *cut_mark; | 
					
						
							|  |  |  | #endif /* TABLING_INNER_CUTS */
 | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   int pbvars; | 
					
						
							|  |  |  | #endif /* DEBUG */
 | 
					
						
							|  |  |  |   int nvars; | 
					
						
							|  |  |  |   UInt labelno; | 
					
						
							|  |  |  |   int or_found; | 
					
						
							|  |  |  |   UInt max_args; | 
					
						
							|  |  |  |   int MaxCTemps; | 
					
						
							|  |  |  |   UInt tmpreg; | 
					
						
							|  |  |  |   Int vreg; | 
					
						
							|  |  |  |   Int vadr; | 
					
						
							|  |  |  |   Int *Uses; | 
					
						
							|  |  |  |   Term *Contents; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |   int needs_env; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   CIntermediates cint; | 
					
						
							|  |  |  | } compiler_struct; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 17:15:04 -05:00
										 |  |  | static int active_branch(int, int); | 
					
						
							|  |  |  | static void c_var(Term, Int, unsigned int, unsigned int, compiler_struct *); | 
					
						
							|  |  |  | static void reset_vars(Ventry *); | 
					
						
							|  |  |  | static Term optimize_ce(Term, unsigned int, unsigned int, compiler_struct *); | 
					
						
							|  |  |  | static void c_arg(Int, Term, unsigned int, unsigned int, compiler_struct *); | 
					
						
							|  |  |  | static void c_args(Term, unsigned int, compiler_struct *); | 
					
						
							|  |  |  | static void c_eq(Term, Term, compiler_struct *); | 
					
						
							|  |  |  | static void c_test(Int, Term, compiler_struct *); | 
					
						
							|  |  |  | static void c_bifun(basic_preds, Term, Term, Term, Term, Term, compiler_struct *); | 
					
						
							|  |  |  | static void c_goal(Term, Term, compiler_struct *); | 
					
						
							|  |  |  | static void c_body(Term, Term, compiler_struct *); | 
					
						
							|  |  |  | static void c_head(Term, compiler_struct *); | 
					
						
							|  |  |  | static int usesvar(compiler_vm_op); | 
					
						
							|  |  |  | static CELL *init_bvarray(int, compiler_struct *); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #ifdef DEBUG
 | 
					
						
							| 
									
										
										
										
											2013-04-25 17:15:04 -05:00
										 |  |  | static void clear_bvarray(int, CELL *, compiler_struct *); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2013-04-25 17:15:04 -05:00
										 |  |  | static void clear_bvarray(int, CELL *); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-04-25 17:15:04 -05:00
										 |  |  | static void add_bvarray_op(PInstr *,CELL *, int, compiler_struct *); | 
					
						
							|  |  |  | static void AssignPerm(PInstr *, compiler_struct *); | 
					
						
							|  |  |  | static void CheckUnsafe(PInstr *, compiler_struct *); | 
					
						
							|  |  |  | static void CheckVoids(compiler_struct *); | 
					
						
							|  |  |  | static  int checktemp(Int, Int, compiler_vm_op, compiler_struct *); | 
					
						
							|  |  |  | static  Int checkreg(Int, Int, compiler_vm_op, int, compiler_struct *); | 
					
						
							|  |  |  | static void c_layout(compiler_struct *); | 
					
						
							|  |  |  | static void c_optimize(PInstr *); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #ifdef SFUNC
 | 
					
						
							| 
									
										
										
										
											2013-04-25 17:15:04 -05:00
										 |  |  | static void compile_sf_term(Term, int); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | push_branch(int id, Term cmvar, compiler_struct *cglobs) { | 
					
						
							|  |  |  |   cglobs->branch_pointer->id = id; | 
					
						
							|  |  |  |   cglobs->branch_pointer->cm = cmvar; | 
					
						
							|  |  |  |   cglobs->branch_pointer++; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | pop_branch(compiler_struct *cglobs) { | 
					
						
							|  |  |  |   cglobs->branch_pointer--; | 
					
						
							|  |  |  |   return(cglobs->branch_pointer->id); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef TABLING
 | 
					
						
							|  |  |  | #define is_tabled(pe)   (pe->PredFlags & TabledPredFlag)
 | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int  | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | active_branch(int i, int onbranch) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   /*  register int *bp;*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (i == onbranch); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   /*  bp = cglobs->branch_pointer;
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   while (bp > parent_branches) { | 
					
						
							|  |  |  |     if (*--bp == onbranch) | 
					
						
							|  |  |  |       return (TRUE); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return(i==onbranch);*/ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | #define FAIL(M,T,E) { LOCAL_ErrorMessage=M; LOCAL_Error_TYPE = T; LOCAL_Error_Term = E; return; }
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-31 15:51:18 +01:00
										 |  |  | #if USE_SYSTEM_MALLOC
 | 
					
						
							|  |  |  | #define IsNewVar(v) ((CELL *)(v) >= H0  && (CELL *)(v) < LCL0)
 | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | #define IsNewVar(v) (Addr(v)<cglobs->cint.freep0 || Addr(v)>cglobs->cint.freep)
 | 
					
						
							| 
									
										
										
										
											2010-03-31 15:51:18 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | inline static void pop_code(unsigned int, compiler_struct *); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | inline static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | pop_code(unsigned int level, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   if (level == 0) | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   if (cglobs->cint.cpc->op == pop_op) | 
					
						
							|  |  |  |     ++(cglobs->cint.cpc->rnd1); | 
					
						
							| 
									
										
										
										
											2003-05-19 13:04:09 +00:00
										 |  |  |   else { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_emit(pop_op, One, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2003-05-19 13:04:09 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | adjust_current_commits(compiler_struct *cglobs) { | 
					
						
							|  |  |  |   branch *bp = cglobs->branch_pointer; | 
					
						
							|  |  |  |   while (bp > cglobs->parent_branches) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     bp--; | 
					
						
							|  |  |  |     if (bp->cm != TermNil) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       c_var(bp->cm, patch_b_flag, 1, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | check_var(Term t, unsigned int level, Int argno, compiler_struct *cglobs) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   int flags, new = FALSE; | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |   Ventry *v = (Ventry *)t; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (IsNewVar(v)) {		/* new var */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     v = (Ventry *) Yap_AllocCMem(sizeof(*v), &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2011-03-30 15:32:59 +01:00
										 |  |  | #if YAPOR_SBA
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     v->SelfOfVE = 0; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     v->SelfOfVE = (CELL) v; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     v->AdrsOfVE = t; | 
					
						
							|  |  |  |     *CellPtr(t) = (CELL) v; | 
					
						
							|  |  |  |     v->KindOfVE = v->NoOfVE = Unassigned; | 
					
						
							|  |  |  |     flags = 0; | 
					
						
							|  |  |  |     /* Be careful with eithers. I may make a variable global in a branch,
 | 
					
						
							|  |  |  |        and not in another. | 
					
						
							|  |  |  |        a :- (b([X]) ; c), go(X). | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |        This variable will not be globalised if we are coming from | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |        the second branch. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        I also need to protect the onhead because Luis uses that to | 
					
						
							|  |  |  |        optimise unification in the body of a clause, eg | 
					
						
							|  |  |  |        a :- (X = 2 ; c), go(X). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        And, yes, there is code like this... | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     if (((level > 0 || cglobs->onhead) && cglobs->curbranch == 0) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	|| argno == save_pair_flag || | 
					
						
							|  |  |  | 	argno == save_appl_flag) | 
					
						
							|  |  |  |       flags |= SafeVar; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     if ((level > 0  && cglobs->curbranch == 0) || argno == save_pair_flag || | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	argno == save_appl_flag) | 
					
						
							|  |  |  |       flags |= GlobalVal; | 
					
						
							|  |  |  |     v->FlagsOfVE = flags; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     v->BranchOfVE = cglobs->onbranch; | 
					
						
							|  |  |  |     v->NextOfVE = cglobs->vtable; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     v->RCountOfVE = 0; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     v->AgeOfVE = v->FirstOfVE = cglobs->goalno; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     new = TRUE; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     cglobs->vtable = v; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     v->FlagsOfVE |= NonVoid; | 
					
						
							|  |  |  |     if (v->BranchOfVE > 0) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       if (!active_branch(v->BranchOfVE, cglobs->onbranch)) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	v->AgeOfVE = v->FirstOfVE = 1; | 
					
						
							|  |  |  | 	new = FALSE; | 
					
						
							|  |  |  | 	v->FlagsOfVE |= BranchVar; | 
					
						
							|  |  |  | 	/* set the original instruction correctly */ | 
					
						
							|  |  |  | 	switch (v->FirstOpForV->op) { | 
					
						
							|  |  |  | 	case get_var_op: | 
					
						
							|  |  |  | 	  v->FirstOpForV->op = get_val_op; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	case unify_var_op: | 
					
						
							|  |  |  | 	  v->FirstOpForV->op = unify_val_op; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	case unify_last_var_op: | 
					
						
							|  |  |  | 	  v->FirstOpForV->op = unify_last_val_op; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	case put_var_op: | 
					
						
							|  |  |  | 	  v->FirstOpForV->op = put_val_op; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	case write_var_op: | 
					
						
							|  |  |  | 	  v->FirstOpForV->op = write_val_op; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   if (cglobs->onhead) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     v->FlagsOfVE |= OnHeadFlag; | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |   return new; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | tag_var(Term t, int new, compiler_struct *cglobs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Ventry *v = (Ventry *) t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (new) { | 
					
						
							|  |  |  |     v->FirstOpForV = cglobs->cint.cpc; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   v->LastOpForV = cglobs->cint.cpc; | 
					
						
							|  |  |  |   ++(v->RCountOfVE); | 
					
						
							|  |  |  |   if (cglobs->onlast) | 
					
						
							|  |  |  |     v->FlagsOfVE |= OnLastGoal; | 
					
						
							|  |  |  |   if (v->AgeOfVE < cglobs->goalno) | 
					
						
							|  |  |  |     v->AgeOfVE = cglobs->goalno; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | c_var(Term t, Int argno, unsigned int arity, unsigned int level, compiler_struct *cglobs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int new = check_var(Deref(t), level, argno, cglobs); | 
					
						
							|  |  |  |   t = Deref(t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   switch (argno) { | 
					
						
							|  |  |  |   case save_b_flag: | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |     Yap_emit(save_b_op, t, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     break; | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  |   case commit_b_flag: | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |     Yap_emit(commit_b_op, t, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_emit(empty_call_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |     Yap_emit(restore_tmps_and_skip_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     break; | 
					
						
							|  |  |  |   case patch_b_flag: | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |     Yap_emit(patch_b_op, t, 0, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     break; | 
					
						
							|  |  |  |   case save_pair_flag: | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |     Yap_emit(save_pair_op, t, 0, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     break; | 
					
						
							|  |  |  |   case save_appl_flag: | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |     Yap_emit(save_appl_op, t, 0, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     break; | 
					
						
							|  |  |  |   case f_flag: | 
					
						
							|  |  |  |     if (new) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       ++cglobs->nvars; | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |       Yap_emit(f_var_op, t, (CELL)arity, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2011-02-14 11:29:20 -08:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |       Yap_emit(f_val_op, t, (CELL)arity, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2011-02-14 11:29:20 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     break; | 
					
						
							|  |  |  |   case bt1_flag: | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |     Yap_emit(fetch_args_for_bccall, t, 0, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     break; | 
					
						
							|  |  |  |   case bt2_flag: | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |     Yap_emit(bccall_op, t, (CELL)cglobs->current_p0, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     break; | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  |     if (argno < 0) { | 
					
						
							|  |  |  |       if (new) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit((cglobs->onhead ? unify_s_var_op : write_s_var_op), v, -argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit((cglobs->onhead ? unify_s_val_op : write_s_val_op), v, -argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } else | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     if (cglobs->onhead) { | 
					
						
							| 
									
										
										
										
											2010-08-02 13:04:30 +01:00
										 |  |  |       cglobs->space_used ++; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       if (level == 0) | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  | 	Yap_emit((new ? (++cglobs->nvars, get_var_op) : get_val_op), t, argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit((new ? (++cglobs->nvars, (argno == (Int)arity ? | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 			       unify_last_var_op : | 
					
						
							|  |  |  | 			       unify_var_op)) : | 
					
						
							|  |  |  | 	      (argno == (Int)arity ? unify_last_val_op : | 
					
						
							|  |  |  | 	       unify_val_op)), | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  | 	     t, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       if (level == 0) | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  | 	Yap_emit((new ? (++cglobs->nvars, put_var_op) : put_val_op), t, argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  | 	Yap_emit((new ? (++cglobs->nvars, write_var_op) : write_val_op), t, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |   tag_var(t, new, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | reset_vars(Ventry *vtable) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   Ventry *v = vtable; | 
					
						
							|  |  |  |   CELL *t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (v != NIL) { | 
					
						
							|  |  |  |     t = (CELL *) v->AdrsOfVE; | 
					
						
							|  |  |  |     RESET_VARIABLE(t); | 
					
						
							|  |  |  |     v = v->NextOfVE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Term | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | optimize_ce(Term t, unsigned int arity, unsigned int level, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |   CExpEntry *p = cglobs->common_exps; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   int cmp = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  |   if (EAM) return t; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-21 20:00:05 +00:00
										 |  |  |   if (IsApplTerm(t) && IsExtensionFunctor(FunctorOfTerm(t))) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     return (t); | 
					
						
							|  |  |  |   while (p != NULL) { | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |     CELL *oldH = HR; | 
					
						
							|  |  |  |     HR = (CELL *)cglobs->cint.freep; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     cmp = Yap_compare_terms(t, (p->TermOfCE)); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |     HR = oldH; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     if (cmp) { | 
					
						
							|  |  |  |       p = p->NextCE; | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (p != NULL) {		/* already there */ | 
					
						
							|  |  |  |     return (p->VarOfCE); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* first occurrence */ | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |   if (cglobs->onbranch || level > 1) { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     return t; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   ++(cglobs->n_common_exps); | 
					
						
							|  |  |  |   p = (CExpEntry *) Yap_AllocCMem(sizeof(CExpEntry), &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   p->TermOfCE = t; | 
					
						
							|  |  |  |   p->VarOfCE = MkVarTerm(); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   if (HR >= (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* oops, too many new variables */ | 
					
						
							|  |  |  |     save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |     siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |   p->NextCE = cglobs->common_exps; | 
					
						
							|  |  |  |   cglobs->common_exps = p; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (IsApplTerm(t)) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_var(p->VarOfCE, save_appl_flag, arity, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   else if (IsPairTerm(t)) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_var(p->VarOfCE, save_pair_flag, arity, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   return (t); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2002-10-28 20:01:53 +00:00
										 |  |  | compile_sf_term(Term t, int argno, int level) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   Functor f = FunctorOfTerm(t); | 
					
						
							|  |  |  |   CELL *p = ArgsOfSFTerm(t) - 1; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   SFEntry *pe = RepSFProp(Yap_GetAProp(NameOfFunctor(f), SFProperty)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   Term nullvalue = pe->NilValue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (level == 0) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_emit((cglobs->onhead ? get_s_f_op : put_s_f_op), f, argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_emit((cglobs->onhead ? unify_s_f_op : write_s_f_op), f, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   ++level; | 
					
						
							|  |  |  |   while ((argno = *++p)) { | 
					
						
							|  |  |  |     t = Derefa(++p); | 
					
						
							|  |  |  |     if (t != nullvalue) { | 
					
						
							|  |  |  |       if (IsAtomicTerm(t)) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit((cglobs->onhead ? unify_s_a_op : write_s_a_op), t, (CELL) argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       else if (!IsVarTerm(t)) { | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	LOCAL_Error_TYPE = INTERNAL_COMPILER_ERROR; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = TermNil; | 
					
						
							|  |  |  | 	LOCAL_ErrorMessage = "illegal argument of soft functor"; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch, COMPILER_ERR_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_var(t, -argno, arity, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   --level; | 
					
						
							|  |  |  |   if (level == 0) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_emit((cglobs->onhead ? get_s_end_op : put_s_end_op), Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_emit((cglobs->onhead ? unify_s_end_op : write_s_end_op), Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | c_args(Term app, unsigned int level, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   Functor f = FunctorOfTerm(app); | 
					
						
							|  |  |  |   unsigned int Arity = ArityOfFunctor(f); | 
					
						
							|  |  |  |   unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (level == 0) { | 
					
						
							|  |  |  |     if (Arity >= MaxTemps) { | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       LOCAL_Error_TYPE = INTERNAL_COMPILER_ERROR; | 
					
						
							|  |  |  |       LOCAL_Error_Term = TermNil; | 
					
						
							|  |  |  |       LOCAL_ErrorMessage = "exceed maximum arity of compiled goal"; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch, COMPILER_ERR_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     if (Arity > cglobs->max_args) | 
					
						
							|  |  |  |       cglobs->max_args = Arity; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   for (i = 1; i <= Arity; ++i) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_arg(i, ArgOfTerm(i, app), Arity, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | try_store_as_dbterm(Term t, Int argno, unsigned int arity, int level, compiler_struct *cglobs) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |   DBTerm *dbt; | 
					
						
							|  |  |  |   int g; | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   CELL *h0 = HR; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   while ((g=Yap_SizeGroundTerm(t,TRUE)) < 0) { | 
					
						
							|  |  |  |     /* oops, too deep a term */ | 
					
						
							|  |  |  |     save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_Size = 0; | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |     siglongjmp(cglobs->cint.CompilerBotch, OUT_OF_AUX_BOTCH); | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (g < 16) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   /* store ground term away */ | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   HR = CellPtr(cglobs->cint.freep); | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |   if ((dbt = Yap_StoreTermInDB(t, -1)) == NULL) { | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |     HR = h0; | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     switch(LOCAL_Error_TYPE) { | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |     case OUT_OF_STACK_ERROR: | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       LOCAL_Error_TYPE = YAP_NO_ERROR; | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_STACK_BOTCH); | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |     case OUT_OF_TRAIL_ERROR: | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       LOCAL_Error_TYPE = YAP_NO_ERROR; | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TRAIL_BOTCH); | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |     case OUT_OF_HEAP_ERROR: | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       LOCAL_Error_TYPE = YAP_NO_ERROR; | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_HEAP_BOTCH); | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |     case OUT_OF_AUXSPACE_ERROR: | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       LOCAL_Error_TYPE = YAP_NO_ERROR; | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_AUX_BOTCH); | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |     default: | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch,COMPILER_ERR_BOTCH); | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   HR = h0; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |   if (level == 0) | 
					
						
							|  |  |  |     Yap_emit((cglobs->onhead ? get_dbterm_op : put_dbterm_op), dbt->Entry, argno, &cglobs->cint); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     Yap_emit((cglobs->onhead ? (argno == (Int)arity ? unify_last_dbterm_op | 
					
						
							|  |  |  | 				: unify_dbterm_op) : | 
					
						
							|  |  |  | 	      write_dbterm_op), dbt->Entry, Zero, &cglobs->cint); | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | c_arg(Int argno, Term t, unsigned int arity, unsigned int level, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-10-28 20:01:53 +00:00
										 |  |  |  restart: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (IsVarTerm(t)) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_var(t, argno, arity, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   else if (IsAtomTerm(t)) { | 
					
						
							| 
									
										
										
										
											2008-12-09 14:23:19 +00:00
										 |  |  |     if (level == 0) {       | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? get_atom_op : put_atom_op), (CELL) t, argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2008-12-09 14:23:19 +00:00
										 |  |  |     } else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? (argno == (Int)arity ? unify_last_atom_op | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 		      : unify_atom_op) : | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    write_atom_op), (CELL) t, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  |   } else  if (IsIntegerTerm(t) || IsFloatTerm(t) || IsBigIntTerm(t) || IsStringTerm(t)) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     if (!IsIntTerm(t)) { | 
					
						
							|  |  |  |       if (IsFloatTerm(t)) { | 
					
						
							|  |  |  | 	if (level == 0) | 
					
						
							| 
									
										
										
										
											2006-09-20 20:03:51 +00:00
										 |  |  | 	  Yap_emit((cglobs->onhead ? get_float_op : put_float_op), t, argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit((cglobs->onhead ? (argno == (Int)arity ? unify_last_float_op | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 			  : unify_float_op) : | 
					
						
							| 
									
										
										
										
											2006-09-20 20:03:51 +00:00
										 |  |  | 		write_float_op), t, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       } else if (IsLongIntTerm(t)) { | 
					
						
							|  |  |  | 	if (level == 0) | 
					
						
							|  |  |  | 	  Yap_emit((cglobs->onhead ? get_longint_op : put_longint_op), t, argno, &cglobs->cint); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	  Yap_emit((cglobs->onhead ? (argno == (Int)arity ? unify_last_longint_op | 
					
						
							|  |  |  | 			  : unify_longint_op) : | 
					
						
							|  |  |  | 		write_longint_op), t, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  |       } else if (IsStringTerm(t)) { | 
					
						
							|  |  |  | 	/* we are taking a string, that is supposed to be
 | 
					
						
							|  |  |  | 	 guarded in the clause itself. . */ | 
					
						
							|  |  |  | 	CELL l1 = ++cglobs->labelno; | 
					
						
							|  |  |  | 	CELL *src = RepAppl(t); | 
					
						
							|  |  |  | 	PInstr *ocpc = cglobs->cint.cpc, *OCodeStart = cglobs->cint.CodeStart; | 
					
						
							|  |  |  | 	Int sz = (3+src[1])*sizeof(CELL); | 
					
						
							|  |  |  | 	CELL *dest; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* use a special list to store the blobs */ | 
					
						
							|  |  |  | 	cglobs->cint.cpc = cglobs->cint.icpc; | 
					
						
							|  |  |  | 	/*      if (IsFloatTerm(t)) {
 | 
					
						
							|  |  |  | 		Yap_emit(align_float_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 		}*/ | 
					
						
							|  |  |  | 	Yap_emit(label_op, l1, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	dest =  | 
					
						
							|  |  |  | 	  Yap_emit_extra_size(blob_op, sz/CellSize, sz, &cglobs->cint); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* copy the bignum */ | 
					
						
							|  |  |  | 	memcpy(dest, src, sz); | 
					
						
							|  |  |  | 	/* note that we don't need to copy size info, unless we wanted
 | 
					
						
							|  |  |  | 	 to garbage collect clauses ;-) */ | 
					
						
							|  |  |  | 	cglobs->cint.icpc = cglobs->cint.cpc; | 
					
						
							|  |  |  | 	if (cglobs->cint.BlobsStart == NULL) | 
					
						
							|  |  |  | 	  cglobs->cint.BlobsStart = cglobs->cint.CodeStart; | 
					
						
							|  |  |  | 	cglobs->cint.cpc = ocpc; | 
					
						
							|  |  |  | 	cglobs->cint.CodeStart = OCodeStart; | 
					
						
							|  |  |  | 	/* The argument to pass to the structure is now the label for
 | 
					
						
							|  |  |  | 	   where we are storing the blob */ | 
					
						
							|  |  |  | 	if (level == 0) | 
					
						
							| 
									
										
										
										
											2013-12-04 23:01:30 +00:00
										 |  |  | 	  Yap_emit((cglobs->onhead ? get_string_op : put_string_op), l1, argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | 	  Yap_emit((cglobs->onhead ? (argno == (Int)arity ? unify_last_string_op | 
					
						
							|  |  |  | 			  : unify_string_op) : | 
					
						
							| 
									
										
										
										
											2013-12-05 21:26:46 +00:00
										 |  |  | 		write_string_op), l1, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2006-09-20 20:03:51 +00:00
										 |  |  |       } else { | 
					
						
							|  |  |  | 	/* we are taking a blob, that is a binary that is supposed to be
 | 
					
						
							|  |  |  | 	 guarded in the clause itself. Possible examples include | 
					
						
							|  |  |  | 	 floats, long ints, bignums, bitmaps.... */ | 
					
						
							|  |  |  | 	CELL l1 = ++cglobs->labelno; | 
					
						
							|  |  |  | 	CELL *src = RepAppl(t); | 
					
						
							|  |  |  | 	PInstr *ocpc = cglobs->cint.cpc, *OCodeStart = cglobs->cint.CodeStart; | 
					
						
							| 
									
										
										
										
											2011-06-21 15:07:54 +01:00
										 |  |  | 	Int sz = 2*sizeof(CELL)+sizeof(Functor)+ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  sizeof(MP_INT)+ | 
					
						
							| 
									
										
										
										
											2009-02-09 21:56:40 +00:00
										 |  |  | 	   ((((MP_INT *)(RepAppl(t)+2))->_mp_alloc)*sizeof(mp_limb_t)); | 
					
						
							| 
									
										
										
										
											2006-09-20 20:03:51 +00:00
										 |  |  | 	CELL *dest; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* use a special list to store the blobs */ | 
					
						
							|  |  |  | 	cglobs->cint.cpc = cglobs->cint.icpc; | 
					
						
							|  |  |  | 	/*      if (IsFloatTerm(t)) {
 | 
					
						
							|  |  |  | 		Yap_emit(align_float_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 		}*/ | 
					
						
							|  |  |  | 	Yap_emit(label_op, l1, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	dest =  | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit_extra_size(blob_op, sz/CellSize, sz, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2006-09-20 20:03:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* copy the bignum */ | 
					
						
							|  |  |  | 	memcpy(dest, src, sz); | 
					
						
							|  |  |  | 	/* note that we don't need to copy size info, unless we wanted
 | 
					
						
							|  |  |  | 	 to garbage collect clauses ;-) */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	cglobs->cint.icpc = cglobs->cint.cpc; | 
					
						
							|  |  |  | 	if (cglobs->cint.BlobsStart == NULL) | 
					
						
							|  |  |  | 	  cglobs->cint.BlobsStart = cglobs->cint.CodeStart; | 
					
						
							|  |  |  | 	cglobs->cint.cpc = ocpc; | 
					
						
							|  |  |  | 	cglobs->cint.CodeStart = OCodeStart; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* The argument to pass to the structure is now the label for
 | 
					
						
							|  |  |  | 	   where we are storing the blob */ | 
					
						
							|  |  |  | 	if (level == 0) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit((cglobs->onhead ? get_bigint_op : put_bigint_op), l1, argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit((cglobs->onhead ? (argno == (Int)arity ? unify_last_bigint_op | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 			  : unify_bigint_op) : | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 		write_bigint_op), l1, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       /* That's it folks! */ | 
					
						
							|  |  |  |       return; | 
					
						
							| 
									
										
										
										
											2005-05-12 03:36:33 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     if (level == 0) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? get_num_op : put_num_op), (CELL) t, argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? (argno == (Int)arity ? unify_last_num_op | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 		      : unify_num_op) : | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    write_num_op), (CELL) t, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2005-05-12 03:36:33 +00:00
										 |  |  |   } else if (IsPairTerm(t)) { | 
					
						
							| 
									
										
										
										
											2010-08-02 13:04:30 +01:00
										 |  |  |     cglobs->space_used += 2; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     if (optimizer_on && level < 6) { | 
					
						
							| 
									
										
										
										
											2011-07-22 01:35:43 -07:00
										 |  |  | #if !defined(THREADS) && !defined(YAPOR)
 | 
					
						
							| 
									
										
										
										
											2008-08-06 17:32:22 +00:00
										 |  |  |       /* discard code sharing because we cannot write on shared stuff */ | 
					
						
							| 
									
										
										
										
											2011-08-31 13:58:06 -07:00
										 |  |  |       if (FALSE && !(cglobs->cint.CurrentPred->PredFlags & (DynamicPredFlag|LogUpdatePredFlag))) { | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | 	if (try_store_as_dbterm(t, argno, arity, level, cglobs)) | 
					
						
							|  |  |  | 	  return; | 
					
						
							| 
									
										
										
										
											2008-08-06 17:32:22 +00:00
										 |  |  |       }      | 
					
						
							|  |  |  | #endif 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       t = optimize_ce(t, arity, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       if (IsVarTerm(t)) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_var(t, argno, arity, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (level == 0) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? get_list_op : put_list_op), Zero, argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     else if (argno == (Int)arity) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? unify_last_list_op : write_last_list_op), Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? unify_list_op : write_list_op), Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     ++level; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_arg(1, HeadOfTerm(t), 2, level, cglobs); | 
					
						
							| 
									
										
										
										
											2002-10-28 20:01:53 +00:00
										 |  |  |     if (argno == (Int)arity) { | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |      /* optimise for tail recursion */ | 
					
						
							| 
									
										
										
										
											2002-10-28 20:01:53 +00:00
										 |  |  |       t = TailOfTerm(t); | 
					
						
							|  |  |  |       goto restart; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_arg(2, TailOfTerm(t), 2, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     --level; | 
					
						
							| 
									
										
										
										
											2003-05-19 13:04:09 +00:00
										 |  |  |     if (argno != (Int)arity) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       pop_code(level, cglobs); | 
					
						
							| 
									
										
										
										
											2003-05-19 13:04:09 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } else if (IsRefTerm(t)) { | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  |     PELOCK(40,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     if (!(cglobs->cint.CurrentPred->PredFlags & (DynamicPredFlag|LogUpdatePredFlag))) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |       CACHE_REGS | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  |       UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       FAIL("can not compile data base reference",TYPE_ERROR_CALLABLE,t); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  |       UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2005-05-12 03:36:33 +00:00
										 |  |  |       cglobs->hasdbrefs = TRUE; | 
					
						
							|  |  |  |       if (level == 0) | 
					
						
							|  |  |  | 	Yap_emit((cglobs->onhead ? get_atom_op : put_atom_op), (CELL) t, argno, &cglobs->cint);       | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	Yap_emit((cglobs->onhead ? (argno == (Int)arity ? unify_last_atom_op | 
					
						
							|  |  |  | 				    : unify_atom_op) : | 
					
						
							|  |  |  | 		  write_atom_op), (CELL) t, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  |     if (SFTerm(t)) { | 
					
						
							|  |  |  |       compile_sf_term(t, argno); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     if (optimizer_on) { | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |       if (!(cglobs->cint.CurrentPred->PredFlags & (DynamicPredFlag|LogUpdatePredFlag))) { | 
					
						
							|  |  |  | 	if (try_store_as_dbterm(t, argno, arity, level, cglobs)) | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  |       }       | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       t = optimize_ce(t, arity, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       if (IsVarTerm(t)) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_var(t, argno, arity, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-08-02 13:04:30 +01:00
										 |  |  |     cglobs->space_used += 1+arity; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     if (level == 0) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? get_struct_op : put_struct_op), | 
					
						
							|  |  |  | 	   (CELL) FunctorOfTerm(t), argno, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     else if (argno == (Int)arity) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? unify_last_struct_op : write_last_struct_op), | 
					
						
							|  |  |  | 	   (CELL) FunctorOfTerm(t), Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit((cglobs->onhead ? unify_struct_op : write_struct_op), | 
					
						
							|  |  |  | 	   (CELL) FunctorOfTerm(t), Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     ++level; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_args(t, level, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     --level; | 
					
						
							| 
									
										
										
										
											2003-05-19 13:04:09 +00:00
										 |  |  |     if (argno != (Int)arity) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       pop_code(level, cglobs); | 
					
						
							| 
									
										
										
										
											2003-05-19 13:04:09 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | c_eq(Term t1, Term t2, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2006-04-05 00:16:55 +00:00
										 |  |  |   if (t1 == t2) { | 
					
						
							|  |  |  |     Yap_emit(nop_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |   if (IsNonVarTerm(t1)) { | 
					
						
							|  |  |  |     if (IsVarTerm(t2)) { | 
					
						
							|  |  |  |       Term t = t1; | 
					
						
							|  |  |  |       t1 = t2; | 
					
						
							|  |  |  |       t2 = t; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       /* compile unification */ | 
					
						
							|  |  |  |       if (IsAtomicTerm(t1)) { | 
					
						
							|  |  |  | 	/* just check if they unify */ | 
					
						
							|  |  |  | 	if (!IsAtomicTerm(t2) || !Yap_unify(t1,t2)) { | 
					
						
							|  |  |  | 	  /* they don't */ | 
					
						
							|  |  |  | 	  Yap_emit(fail_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* they do */ | 
					
						
							|  |  |  | 	Yap_emit(nop_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  |       } else if (IsPairTerm(t1)) { | 
					
						
							|  |  |  | 	/* just check if they unify */ | 
					
						
							|  |  |  | 	if (!IsPairTerm(t2)) { | 
					
						
							|  |  |  | 	  /* they don't */ | 
					
						
							|  |  |  | 	  Yap_emit(fail_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* they might */ | 
					
						
							|  |  |  | 	c_eq(HeadOfTerm(t1), HeadOfTerm(t2), cglobs); | 
					
						
							|  |  |  | 	c_eq(TailOfTerm(t1), TailOfTerm(t2), cglobs); | 
					
						
							| 
									
										
										
										
											2006-04-05 00:16:55 +00:00
										 |  |  | 	return; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       } else if (IsRefTerm(t1)) { | 
					
						
							|  |  |  | 	/* just check if they unify */ | 
					
						
							|  |  |  | 	if (t1 != t2) { | 
					
						
							|  |  |  | 	  /* they don't */ | 
					
						
							|  |  |  | 	  Yap_emit(fail_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* they do */ | 
					
						
							|  |  |  | 	Yap_emit(nop_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2006-04-05 00:16:55 +00:00
										 |  |  | 	return; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       } else { | 
					
						
							|  |  |  | 	/* compound terms */ | 
					
						
							|  |  |  | 	Functor f = FunctorOfTerm(t1); | 
					
						
							|  |  |  | 	UInt i, max; | 
					
						
							|  |  |  | 	/* just check if they unify */ | 
					
						
							|  |  |  | 	if (!IsApplTerm(t2) || | 
					
						
							|  |  |  | 	    FunctorOfTerm(t2) != f) { | 
					
						
							|  |  |  | 	  /* they don't */ | 
					
						
							|  |  |  | 	  Yap_emit(fail_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* they might */ | 
					
						
							|  |  |  | 	max = ArityOfFunctor(f); | 
					
						
							|  |  |  | 	for (i=0; i < max; i++) { | 
					
						
							|  |  |  | 	  c_eq(ArgOfTerm(i+1,t1), ArgOfTerm(i+1,t2), cglobs); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2008-12-09 14:23:19 +00:00
										 |  |  |   /* first argument is an unbound var */ | 
					
						
							| 
									
										
										
										
											2011-07-05 07:03:14 +01:00
										 |  |  |   if (IsNewVar(t1) && !IsVarTerm(t2) && !(cglobs->cint.CurrentPred->PredFlags & TabledPredFlag)) { | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |     Int v; | 
					
						
							| 
									
										
										
										
											2011-07-05 07:03:14 +01:00
										 |  |  |      | 
					
						
							|  |  |  |     v = --cglobs->tmpreg; | 
					
						
							|  |  |  |     c_arg(v, t2, 0, 0, cglobs); | 
					
						
							|  |  |  |     cglobs->onhead = TRUE; | 
					
						
							|  |  |  |     c_var(t1, v, 0, 0, cglobs); | 
					
						
							|  |  |  |     cglobs->onhead = FALSE;     | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2008-12-13 21:22:07 +00:00
										 |  |  |     if (IsVarTerm(t2)) { | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  |       c_var(t1, 0, 0, 0, cglobs); | 
					
						
							|  |  |  |       cglobs->onhead = TRUE; | 
					
						
							|  |  |  |       c_var(t2, 0, 0, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2008-12-13 21:22:07 +00:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  |       Int v = --cglobs->tmpreg; | 
					
						
							|  |  |  |       c_var(t1, v, 0, 0, cglobs); | 
					
						
							|  |  |  |       cglobs->onhead = TRUE; | 
					
						
							| 
									
										
										
										
											2009-02-16 17:15:21 +00:00
										 |  |  |       c_arg(v, t2, 0, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2008-12-13 21:22:07 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     cglobs->onhead = FALSE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | c_test(Int Op, Term t1, compiler_struct *cglobs) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   Term t = Deref(t1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-15 10:30:53 -06:00
										 |  |  |   /* be caareful, has to be first occurrence */ | 
					
						
							|  |  |  |   if (Op == _save_by) { | 
					
						
							|  |  |  |     if (!IsNewVar(t)) { | 
					
						
							|  |  |  | 	char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	LOCAL_Error_TYPE = TYPE_ERROR_VARIABLE; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = t; | 
					
						
							|  |  |  | 	LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							|  |  |  | 	Yap_bip_name(Op, s); | 
					
						
							|  |  |  | 	sprintf(LOCAL_ErrorMessage, "compiling %s/2 on bound variable", s); | 
					
						
							|  |  |  | 	save_machine_regs(); | 
					
						
							|  |  |  | 	siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     c_var(t, save_b_flag, 1, 0, cglobs); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-08-31 23:59:34 -05:00
										 |  |  |   if (!IsVarTerm(t) || IsNewVar(t)) { | 
					
						
							| 
									
										
										
										
											2009-05-24 19:20:00 -05:00
										 |  |  |     Term tn = MkVarTerm(); | 
					
						
							|  |  |  |     c_eq(t, tn, cglobs); | 
					
						
							|  |  |  |     t = tn; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-02-14 11:29:20 -08:00
										 |  |  |   if (Op == _cut_by) | 
					
						
							|  |  |  |     c_var(t, commit_b_flag, 1, 0, cglobs); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     c_var(t, f_flag,(unsigned int)Op, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Arithmetic builtins will be compiled in the form:
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    fetch_args_vv   Xi,Xj | 
					
						
							|  |  |  |    put_val	   Xi,Ri | 
					
						
							|  |  |  |    put_val	   Xj,Rj | 
					
						
							|  |  |  |    put_var	   Xk,Ak | 
					
						
							|  |  |  |    bip_body	   Op,Xk | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The put_var should always be disposable, and the put_vals can be disposed of if R is an X. | 
					
						
							|  |  |  | This, in the best case, Ri and Rj are WAM temp registers and this will reduce to: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    bip		Op,Ak,Ri,Rj | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | meaning a single WAM op will call the clause. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | If one of the arguments is a constant, the result will be: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    fetch_args_vc   Xi,C | 
					
						
							|  |  |  |    put_val	   Xi,Ri | 
					
						
							|  |  |  |    put_var	   Xk,Ak | 
					
						
							|  |  |  |    bip_body	   Op,Xk | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | and this should reduce to : | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bip_cons	   Op,Xk,Ri,C | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2010-02-22 23:08:40 +00:00
										 |  |  | c_bifun(basic_preds Op, Term t1, Term t2, Term t3, Term Goal, Term mod, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							|  |  |  |    /* compile Z = X Op Y  arithmetic function */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* first we fetch the arguments */ | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (IsVarTerm(t1)) { | 
					
						
							| 
									
										
										
										
											2009-05-24 19:20:00 -05:00
										 |  |  |     if (IsVarTerm(t2)) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* first temp */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Int v1 = --cglobs->tmpreg; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* second temp */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Int v2 = --cglobs->tmpreg; | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit(fetch_args_vv_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* these should be the arguments */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_var(t1, v1, 0, 0, cglobs); | 
					
						
							|  |  |  | 	c_var(t2, v2, 0, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* now we know where the arguments are */ | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  |       if (Op == _arg) { | 
					
						
							| 
									
										
										
										
											2006-05-16 18:37:31 +00:00
										 |  |  | 	/* we know the second argument is bound */ | 
					
						
							|  |  |  | 	if (IsPrimitiveTerm(t2) || IsNumTerm(t2)) { | 
					
						
							|  |  |  | 	  Yap_emit(fail_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 	  Term tn = MkVarTerm(); | 
					
						
							|  |  |  | 	  Int v1 = --cglobs->tmpreg; | 
					
						
							|  |  |  | 	  Int v2 = --cglobs->tmpreg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  c_eq(t2, tn, cglobs); | 
					
						
							|  |  |  | 	  Yap_emit(fetch_args_vv_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  /* these should be the arguments */ | 
					
						
							|  |  |  | 	  c_var(t1, v1, 0, 0, cglobs); | 
					
						
							|  |  |  | 	  c_var(tn, v2, 0, 0, cglobs); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |       /* it has to be either an integer or a floating point */ | 
					
						
							| 
									
										
										
										
											2005-04-10 04:01:15 +00:00
										 |  |  |       } else if (IsIntegerTerm(t2)) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* first temp */ | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  | 	Int v1 = 0; | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-06 18:35:05 +00:00
										 |  |  | 	Yap_emit(fetch_args_vi_op, IntegerOfTerm(t2), 0L, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* these should be the arguments */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_var(t1, v1, 0, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* now we know where the arguments are */ | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  | 	char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	LOCAL_Error_TYPE = TYPE_ERROR_NUMBER; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = t2; | 
					
						
							|  |  |  | 	LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	sprintf(LOCAL_ErrorMessage, "compiling %s/2 with output bound", s); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { /* t1 is bound */ | 
					
						
							|  |  |  |     /* it has to be either an integer or a floating point */ | 
					
						
							|  |  |  |     if (IsVarTerm(t2)) { | 
					
						
							|  |  |  |       if (IsNewVar(t2)) { | 
					
						
							|  |  |  | 	char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	LOCAL_Error_TYPE = INSTANTIATION_ERROR; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = t2; | 
					
						
							|  |  |  | 	LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	sprintf(LOCAL_ErrorMessage, "compiling %s/3",s); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |       if (Op == _functor) { | 
					
						
							|  |  |  | 	/* both arguments are bound, we must perform unification */ | 
					
						
							|  |  |  | 	Int i2; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (!IsIntegerTerm(t2)) { | 
					
						
							|  |  |  | 	  char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  LOCAL_Error_TYPE = TYPE_ERROR_INTEGER; | 
					
						
							|  |  |  | 	  LOCAL_Error_Term = t2; | 
					
						
							|  |  |  | 	  LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	  Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  sprintf(LOCAL_ErrorMessage, "compiling functor/3"); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	  siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	if (i2 < 0) { | 
					
						
							|  |  |  | 	  char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  LOCAL_Error_TYPE = DOMAIN_ERROR_NOT_LESS_THAN_ZERO; | 
					
						
							|  |  |  | 	  LOCAL_Error_Term = t2; | 
					
						
							|  |  |  | 	  LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	  Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  sprintf(LOCAL_ErrorMessage, "compiling functor/3"); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	  siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (IsNumTerm(t1)) { | 
					
						
							|  |  |  | 	  /* we will always fail */ | 
					
						
							|  |  |  | 	  if (i2) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    c_goal(MkAtomTerm(AtomFalse), mod, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	} else if (!IsAtomTerm(t1)) { | 
					
						
							|  |  |  | 	  char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  LOCAL_Error_TYPE = TYPE_ERROR_ATOM; | 
					
						
							|  |  |  | 	  LOCAL_Error_Term = t2; | 
					
						
							|  |  |  | 	  LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	  Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  sprintf(LOCAL_ErrorMessage, "compiling functor/3"); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	  siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (i2 == 0) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  c_eq(t1, t3, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	  CELL *hi = HR; | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  Int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (t1 == TermDot && i2 == 2) { | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	    if (HR+2 >= (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	      /* oops, too many new variables */ | 
					
						
							|  |  |  | 	      save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	      siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	    RESET_VARIABLE(HR); | 
					
						
							|  |  |  | 	    RESET_VARIABLE(HR+1); | 
					
						
							|  |  |  | 	    HR += 2; | 
					
						
							|  |  |  | 	    c_eq(AbsPair(HR-2),t3, cglobs); | 
					
						
							| 
									
										
										
										
											2008-12-28 11:00:38 +00:00
										 |  |  | 	  } else if (i2 < 256 && IsAtomTerm(t1)) { | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	    *HR++ = (CELL)Yap_MkFunctor(AtomOfTerm(t1),i2); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	    for (i=0; i < i2; i++) { | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	      if (HR >= (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 		/* oops, too many new variables */ | 
					
						
							|  |  |  | 		save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 		siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	      } | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	      RESET_VARIABLE(HR); | 
					
						
							|  |  |  | 	      HR++;	     | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    c_eq(AbsAppl(hi),t3, cglobs); | 
					
						
							| 
									
										
										
										
											2007-03-27 13:48:51 +00:00
										 |  |  | 	  } else { | 
					
						
							|  |  |  | 	    /* compile as default */ | 
					
						
							|  |  |  | 	    Functor f = FunctorOfTerm(Goal); | 
					
						
							|  |  |  | 	    Prop p0 = PredPropByFunc(f, mod); | 
					
						
							| 
									
										
										
										
											2009-05-24 19:25:26 -05:00
										 |  |  | 	    if (EndOfPAEntr(p0)) { | 
					
						
							|  |  |  | 	      save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	      siglongjmp(cglobs->cint.CompilerBotch, OUT_OF_HEAP_BOTCH); | 
					
						
							| 
									
										
										
										
											2009-05-24 19:25:26 -05:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2007-03-27 13:48:51 +00:00
										 |  |  | 	    c_args(Goal, 0, cglobs); | 
					
						
							|  |  |  | 	    Yap_emit(safe_call_op, (CELL)p0 , Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	    Yap_emit(empty_call_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	    Yap_emit(restore_tmps_and_skip_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	    return; | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } else if (Op == _arg) { | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	Int i1; | 
					
						
							|  |  |  | 	if (IsIntegerTerm(t1)) | 
					
						
							|  |  |  | 	  i1 = IntegerOfTerm(t1); | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 	  char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  LOCAL_Error_TYPE = TYPE_ERROR_INTEGER; | 
					
						
							|  |  |  | 	  LOCAL_Error_Term = t2; | 
					
						
							|  |  |  | 	  LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	  Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  sprintf(LOCAL_ErrorMessage, "compiling %s/2",  s); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	  save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	  siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (IsAtomicTerm(t2) || | 
					
						
							|  |  |  | 	    (IsApplTerm(t2) && IsExtensionFunctor(FunctorOfTerm(t2)))) { | 
					
						
							|  |  |  | 	  char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  LOCAL_Error_TYPE = TYPE_ERROR_COMPOUND; | 
					
						
							|  |  |  | 	  LOCAL_Error_Term = t2; | 
					
						
							|  |  |  | 	  LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	  Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	  sprintf(LOCAL_ErrorMessage, "compiling %s/2",  s); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	  save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	  siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	} else if (IsApplTerm(t2)) { | 
					
						
							|  |  |  | 	  Functor f = FunctorOfTerm(t2); | 
					
						
							|  |  |  | 	  if (i1 < 1 || i1 > ArityOfFunctor(f)) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    c_goal(MkAtomTerm(AtomFalse), mod, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	  } else { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    c_eq(ArgOfTerm(i1, t2), t3, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	} else if (IsPairTerm(t2)) { | 
					
						
							|  |  |  | 	  switch (i1) { | 
					
						
							|  |  |  | 	  case 1: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    c_eq(HeadOfTerm(t2), t3, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	    return; | 
					
						
							|  |  |  | 	  case 2: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    c_eq(TailOfTerm(t2), t3, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	    return; | 
					
						
							|  |  |  | 	  default: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    c_goal(MkAtomTerm(AtomFalse), mod, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	    return; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  | 	char s[32]; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	LOCAL_Error_TYPE = TYPE_ERROR_INTEGER; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = t2; | 
					
						
							|  |  |  | 	LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	sprintf(LOCAL_ErrorMessage, "compiling %s/2",  s); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |     if (Op == _functor) { | 
					
						
							|  |  |  |       if (!IsAtomicTerm(t1)) { | 
					
						
							|  |  |  | 	char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	LOCAL_Error_TYPE = TYPE_ERROR_ATOM; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = t1; | 
					
						
							|  |  |  | 	LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	sprintf(LOCAL_ErrorMessage, "compiling %s/2",  s); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |       } else { | 
					
						
							|  |  |  | 	if (!IsVarTerm(t2)) { | 
					
						
							|  |  |  | 	  Int arity; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* We actually have the term ready, so let's just do the unification now */ | 
					
						
							|  |  |  | 	  if (!IsIntegerTerm(t2)) { | 
					
						
							|  |  |  | 	    char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	    LOCAL_Error_TYPE = TYPE_ERROR_INTEGER; | 
					
						
							|  |  |  | 	    LOCAL_Error_Term = t2; | 
					
						
							|  |  |  | 	    LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	    Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	    sprintf(LOCAL_ErrorMessage, "compiling %s/2",  s); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	    save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	    siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	  arity = IntOfTerm(t2); | 
					
						
							|  |  |  | 	  if (arity < 0) { | 
					
						
							|  |  |  | 	    /* fail straight away */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    Yap_emit(fail_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	  if (arity) { | 
					
						
							|  |  |  | 	    Term tnew; | 
					
						
							|  |  |  | 	    if (!IsAtomTerm(t1)) { | 
					
						
							|  |  |  | 	      char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	      LOCAL_Error_TYPE = TYPE_ERROR_ATOM; | 
					
						
							|  |  |  | 	      LOCAL_Error_Term = t1; | 
					
						
							|  |  |  | 	      LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	      Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	      sprintf(LOCAL_ErrorMessage, "compiling %s/2",  s); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	      save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	      siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	    if (HR+1+arity >= (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	      /* oops, too many new variables */ | 
					
						
							|  |  |  | 	      save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	      siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	    tnew = AbsAppl(HR); | 
					
						
							|  |  |  | 	    *HR++ = (CELL)Yap_MkFunctor(AtomOfTerm(t1),arity); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	    while (arity--) { | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	      RESET_VARIABLE(HR); | 
					
						
							|  |  |  | 	      HR++; | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    c_eq(tnew, t3, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  } else { | 
					
						
							|  |  |  | 	    /* just unify the two arguments */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    c_eq(t1,t3, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 	  /* first temp */ | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  | 	  Int v1 = 0; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit(fetch_args_cv_op, t1, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  /* these should be the arguments */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  c_var(t2, v1, 0, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  /* now we know where the arguments are */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2005-04-10 04:01:15 +00:00
										 |  |  |     } else if (IsIntegerTerm(t1)) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* first temp */ | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |       Int v1 = 0; | 
					
						
							| 
									
										
										
										
											2006-11-06 18:35:05 +00:00
										 |  |  |       Yap_emit(fetch_args_iv_op, IntegerOfTerm(t1), 0L, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* these should be the arguments */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       c_var(t2, v1, 0, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* now we know where the arguments are */ | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       LOCAL_Error_TYPE = TYPE_ERROR_VARIABLE; | 
					
						
							|  |  |  |       LOCAL_Error_Term = t1; | 
					
						
							|  |  |  |       LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       sprintf(LOCAL_ErrorMessage, "compiling %s/2 with output bound",  s); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   }       | 
					
						
							|  |  |  |   /* then we compile the opcode/result */ | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  |   if (!IsVarTerm(t3)) { | 
					
						
							|  |  |  |     if (Op == _arg) { | 
					
						
							|  |  |  |       Term tmpvar = MkVarTerm(); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |       if (HR == (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  | 	/* oops, too many new variables */ | 
					
						
							|  |  |  | 	save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       c_var(tmpvar,f_flag,(unsigned int)Op, 0, cglobs); | 
					
						
							|  |  |  |       c_eq(tmpvar,t3, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       char s[32]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       LOCAL_Error_TYPE = TYPE_ERROR_VARIABLE; | 
					
						
							|  |  |  |       LOCAL_Error_Term = t3; | 
					
						
							|  |  |  |       LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_bip_name(Op, s); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       sprintf(LOCAL_ErrorMessage, "compiling %s/2 with input unbound",  s); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-22 23:28:04 -05:00
										 |  |  |   } else if (IsNewVar(t3) && cglobs->curbranch == 0 && cglobs->cint.CurrentPred->PredFlags & TabledPredFlag) { | 
					
						
							|  |  |  |     Term nv = MkVarTerm(); | 
					
						
							|  |  |  |     c_var(nv,f_flag,(unsigned int)Op, 0, cglobs); | 
					
						
							|  |  |  |     if (Op == _functor) { | 
					
						
							|  |  |  |       Yap_emit(empty_call_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(restore_tmps_and_skip_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* make sure that we first get the true t3, and then bind it to nv. That way it will be confitional */ | 
					
						
							|  |  |  |     c_eq(t3, nv, cglobs); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   } else if (IsNewVar(t3) && cglobs->curbranch == 0 /* otherwise you may have trouble with z(X) :- ( Z is X*2 ; write(Z)) */) { | 
					
						
							|  |  |  |     c_var(t3,f_flag,(unsigned int)Op, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |     if (Op == _functor) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(empty_call_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(restore_tmps_and_skip_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-20 15:48:04 +00:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     /* generate code for a temp and then unify temp with previous variable */  | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |     Yap_emit(f_0_op, 0, (unsigned int)Op, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2011-02-14 06:57:16 -08:00
										 |  |  |     /* I have to do it here, before I do the unification */ | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |     if (Op == _functor) { | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |       Yap_emit(empty_call_op, Zero, (unsigned int)Op, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(restore_tmps_and_skip_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |     cglobs->onhead = TRUE; | 
					
						
							|  |  |  |     c_var(t3, 0, 0, 0, cglobs); | 
					
						
							|  |  |  |     cglobs->onhead = FALSE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2009-04-07 15:44:46 +01:00
										 |  |  | c_functor(Term Goal, Term mod, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |   Term t1 = ArgOfTerm(1, Goal); | 
					
						
							|  |  |  |   Term t2 = ArgOfTerm(2, Goal); | 
					
						
							|  |  |  |   Term t3 = ArgOfTerm(3, Goal); | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |   if (IsVarTerm(t1) && IsNewVar(t1)) { | 
					
						
							| 
									
										
										
										
											2007-03-27 13:48:51 +00:00
										 |  |  |     c_bifun(_functor, t2, t3, t1, Goal, mod, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |   } else if (IsNonVarTerm(t1)) { | 
					
						
							|  |  |  |     /* just split the structure */ | 
					
						
							|  |  |  |     if (IsAtomicTerm(t1)) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       c_eq(t1,t2, cglobs); | 
					
						
							|  |  |  |       c_eq(t3,MkIntTerm(0), cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |     } else if (IsApplTerm(t1)) { | 
					
						
							|  |  |  |       Functor f = FunctorOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       c_eq(t2,MkAtomTerm(NameOfFunctor(f)), cglobs); | 
					
						
							|  |  |  |       c_eq(t3,MkIntegerTerm(ArityOfFunctor(f)), cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |     } else /* list */ { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       c_eq(t2,TermDot, cglobs); | 
					
						
							|  |  |  |       c_eq(t3,MkIntTerm(2), cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } else if (IsVarTerm(t2) && IsNewVar(t2) && | 
					
						
							|  |  |  | 	     IsVarTerm(t3) && IsNewVar(t3)) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Int v1 = --cglobs->tmpreg; | 
					
						
							| 
									
										
										
										
											2006-11-06 18:35:05 +00:00
										 |  |  |     Yap_emit(fetch_args_vi_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_var(t1, v1, 0, 0, cglobs); | 
					
						
							|  |  |  |     c_var(t2,f_flag,(unsigned int)_functor, 0, cglobs); | 
					
						
							|  |  |  |     c_var(t3,f_flag,(unsigned int)_functor, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     Functor f = FunctorOfTerm(Goal); | 
					
						
							| 
									
										
										
										
											2001-11-15 00:01:43 +00:00
										 |  |  |     Prop p0 = PredPropByFunc(f, mod); | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-24 19:25:26 -05:00
										 |  |  |     if (EndOfPAEntr(p0)) { | 
					
						
							|  |  |  |       save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch, OUT_OF_HEAP_BOTCH); | 
					
						
							| 
									
										
										
										
											2009-05-24 19:25:26 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-06-22 12:40:55 -05:00
										 |  |  |     if (profiling) | 
					
						
							|  |  |  |       Yap_emit(enter_profiling_op, (CELL)RepPredProp(p0), Zero, &cglobs->cint); | 
					
						
							|  |  |  |     else if (call_counting) | 
					
						
							|  |  |  |       Yap_emit(count_call_op, (CELL)RepPredProp(p0), Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_args(Goal, 0, cglobs); | 
					
						
							|  |  |  |     Yap_emit(safe_call_op, (CELL)p0 , Zero, &cglobs->cint); | 
					
						
							|  |  |  |     Yap_emit(empty_call_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |     Yap_emit(restore_tmps_and_skip_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-07 19:56:02 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | IsTrueGoal(Term t) { | 
					
						
							|  |  |  |   if (IsVarTerm(t)) return(FALSE); | 
					
						
							|  |  |  |   if (IsApplTerm(t)) { | 
					
						
							|  |  |  |     Functor f = FunctorOfTerm(t); | 
					
						
							|  |  |  |     if (f == FunctorModule) { | 
					
						
							|  |  |  |       return(IsTrueGoal(ArgOfTerm(2,t))); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-08-01 13:14:17 +00:00
										 |  |  |     if (f == FunctorComma || f == FunctorOr || f == FunctorVBar || f == FunctorArrow) { | 
					
						
							| 
									
										
										
										
											2001-05-07 19:56:02 +00:00
										 |  |  |       return(IsTrueGoal(ArgOfTerm(1,t)) && IsTrueGoal(ArgOfTerm(2,t))); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return(FALSE); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return(t == MkAtomTerm(AtomTrue)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | emit_special_label(Term Goal, compiler_struct *cglobs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   special_label_op lab_op = IntOfTerm(ArgOfTerm(1,Goal)); | 
					
						
							|  |  |  |   special_label_id lab_id = IntOfTerm(ArgOfTerm(2,Goal)); | 
					
						
							|  |  |  |   UInt label_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (lab_op) { | 
					
						
							|  |  |  |   case SPECIAL_LABEL_INIT: | 
					
						
							|  |  |  |     label_name = ++cglobs->labelno; | 
					
						
							|  |  |  |     switch (lab_id) { | 
					
						
							|  |  |  |     case SPECIAL_LABEL_EXCEPTION: | 
					
						
							|  |  |  |       cglobs->cint.exception_handler = label_name; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SPECIAL_LABEL_SUCCESS: | 
					
						
							|  |  |  |       cglobs->cint.success_handler = label_name; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SPECIAL_LABEL_FAILURE: | 
					
						
							|  |  |  |       cglobs->cint.failure_handler = label_name; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-12-09 14:23:19 +00:00
										 |  |  |     Yap_emit_3ops(label_ctl_op, lab_op, lab_id, label_name, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |     break; | 
					
						
							|  |  |  |   case SPECIAL_LABEL_SET: | 
					
						
							|  |  |  |     switch (lab_id) { | 
					
						
							|  |  |  |     case SPECIAL_LABEL_EXCEPTION: | 
					
						
							|  |  |  |       Yap_emit(label_op, cglobs->cint.exception_handler, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SPECIAL_LABEL_SUCCESS: | 
					
						
							|  |  |  |       Yap_emit(label_op, cglobs->cint.success_handler, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SPECIAL_LABEL_FAILURE: | 
					
						
							|  |  |  |       Yap_emit(label_op, cglobs->cint.failure_handler, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case SPECIAL_LABEL_CLEAR: | 
					
						
							| 
									
										
										
										
											2009-02-09 21:56:40 +00:00
										 |  |  |     switch (lab_id) { | 
					
						
							|  |  |  |     case SPECIAL_LABEL_EXCEPTION: | 
					
						
							|  |  |  |       cglobs->cint.exception_handler = 0L; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SPECIAL_LABEL_SUCCESS: | 
					
						
							|  |  |  |       cglobs->cint.success_handler = 0L; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SPECIAL_LABEL_FAILURE: | 
					
						
							|  |  |  |       cglobs->cint.failure_handler = 0L; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2009-04-07 15:44:46 +01:00
										 |  |  | c_goal(Term Goal, Term mod, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   Functor f; | 
					
						
							|  |  |  |   PredEntry *p; | 
					
						
							|  |  |  |   Prop p0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(Goal)) { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Goal = Yap_MkApplTerm(FunctorCall, 1, &Goal); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-03-04 03:25:41 +00:00
										 |  |  |   if (IsApplTerm(Goal) && FunctorOfTerm(Goal) == FunctorModule) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     Term M = ArgOfTerm(1, Goal); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-20 05:17:26 +00:00
										 |  |  |     if (IsVarTerm(M) || !IsAtomTerm(M)) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |       CACHE_REGS | 
					
						
							| 
									
										
										
										
											2002-11-20 05:17:26 +00:00
										 |  |  |       if (IsVarTerm(M)) {	 | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	LOCAL_Error_TYPE = INSTANTIATION_ERROR; | 
					
						
							| 
									
										
										
										
											2002-11-20 05:17:26 +00:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	LOCAL_Error_TYPE = TYPE_ERROR_ATOM; | 
					
						
							| 
									
										
										
										
											2002-11-20 05:17:26 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       LOCAL_Error_Term = M; | 
					
						
							|  |  |  |       LOCAL_ErrorMessage = "in module name"; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch, COMPILER_ERR_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     Goal = ArgOfTerm(2, Goal); | 
					
						
							| 
									
										
										
										
											2004-02-12 12:37:12 +00:00
										 |  |  |     mod = M; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-11-20 05:17:26 +00:00
										 |  |  |   if (IsVarTerm(Goal)) { | 
					
						
							|  |  |  |     Goal = Yap_MkApplTerm(FunctorCall, 1, &Goal); | 
					
						
							|  |  |  |   } else if (IsNumTerm(Goal)) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |     CACHE_REGS | 
					
						
							| 
									
										
										
										
											2002-03-04 03:25:41 +00:00
										 |  |  |     FAIL("goal can not be a number", TYPE_ERROR_CALLABLE, Goal); | 
					
						
							| 
									
										
										
										
											2002-11-20 05:17:26 +00:00
										 |  |  |   } else if (IsRefTerm(Goal)) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |     CACHE_REGS | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_TYPE = TYPE_ERROR_DBREF; | 
					
						
							|  |  |  |     LOCAL_Error_Term = Goal; | 
					
						
							| 
									
										
										
										
											2002-03-04 03:25:41 +00:00
										 |  |  |     FAIL("goal argument in static procedure can not be a data base reference", TYPE_ERROR_CALLABLE, Goal); | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   else if (IsPairTerm(Goal)) { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Goal = Yap_MkApplTerm(FunctorCall, 1, &Goal); | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (IsAtomTerm(Goal)) { | 
					
						
							|  |  |  |     Atom atom = AtomOfTerm(Goal); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (atom == AtomFail || atom == AtomFalse) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(fail_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (atom == AtomTrue || atom == AtomOtherwise) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       if (cglobs->onlast) { | 
					
						
							|  |  |  | 	Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	PELOCK(41,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (is_tabled(cglobs->cint.CurrentPred)) | 
					
						
							|  |  |  | 	  Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2005-03-04 20:30:14 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2005-03-04 20:30:14 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (atom == AtomCut) { | 
					
						
							| 
									
										
										
										
											2009-06-22 12:40:55 -05:00
										 |  |  |       if (profiling) | 
					
						
							|  |  |  | 	Yap_emit(enter_profiling_op, (CELL)RepPredProp(PredPropByAtom(AtomCut,0)), Zero, &cglobs->cint); | 
					
						
							|  |  |  |       else if (call_counting) | 
					
						
							|  |  |  | 	Yap_emit(count_call_op, (CELL)RepPredProp(PredPropByAtom(AtomCut,0)), Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       if (cglobs->onlast) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* never a problem here with a -> b, !, c ; d */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	PELOCK(42,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (is_tabled(cglobs->cint.CurrentPred)) { | 
					
						
							|  |  |  | 	  Yap_emit_3ops(cut_op, Zero, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2010-12-29 09:29:42 -06:00
										 |  |  | 	  /* needs to adjust previous commits */ | 
					
						
							|  |  |  | 	  Yap_emit(empty_call_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  Yap_emit(restore_tmps_and_skip_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* TABLING */
 | 
					
						
							|  |  |  | 	  { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    Yap_emit_3ops(cutexit_op, Zero, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2010-12-29 09:29:42 -06:00
										 |  |  | 	    /* needs to adjust previous commits */ | 
					
						
							|  |  |  | 	    Yap_emit(empty_call_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	    Yap_emit(restore_tmps_and_skip_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	    Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  } | 
					
						
							|  |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit_3ops(cut_op, Zero, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* needs to adjust previous commits */ | 
					
						
							| 
									
										
										
										
											2010-12-29 09:29:42 -06:00
										 |  |  | 	Yap_emit(empty_call_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	Yap_emit(restore_tmps_and_skip_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	adjust_current_commits(cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #ifndef YAPOR
 | 
					
						
							|  |  |  |     else if (atom == AtomRepeat) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       CELL l1 = ++cglobs->labelno; | 
					
						
							|  |  |  |       CELL l2 = ++cglobs->labelno; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       /* I need an either_me */ | 
					
						
							|  |  |  |       cglobs->needs_env = TRUE; | 
					
						
							| 
									
										
										
										
											2009-06-22 12:40:55 -05:00
										 |  |  |       if (profiling) | 
					
						
							|  |  |  | 	Yap_emit(enter_profiling_op, (CELL)RepPredProp(PredPropByAtom(AtomRepeat,0)), Zero, &cglobs->cint); | 
					
						
							|  |  |  |       else if (call_counting) | 
					
						
							|  |  |  | 	Yap_emit(count_call_op, (CELL)RepPredProp(PredPropByAtom(AtomRepeat,0)), Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs->or_found = TRUE; | 
					
						
							|  |  |  |       push_branch(cglobs->onbranch, TermNil, cglobs); | 
					
						
							|  |  |  |       cglobs->curbranch++; | 
					
						
							|  |  |  |       cglobs->onbranch = cglobs->curbranch; | 
					
						
							|  |  |  |       if (cglobs->onlast) | 
					
						
							|  |  |  | 	Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit_3ops(push_or_op, l1, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit_3ops(either_op, l1, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(restore_tmps_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(jump_op, l2, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(label_op, l1, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(pushpop_or_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit_3ops(orelse_op, l1, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(label_op, l2, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       if (cglobs->onlast) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	PELOCK(43,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (is_tabled(cglobs->cint.CurrentPred)) | 
					
						
							|  |  |  | 	  Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	++cglobs->goalno; | 
					
						
							|  |  |  |       cglobs->onbranch = pop_branch(cglobs); | 
					
						
							|  |  |  |       Yap_emit(pop_or_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       /*                      --cglobs->onbranch; */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif /* YAPOR */
 | 
					
						
							| 
									
										
										
										
											2004-02-11 01:20:56 +00:00
										 |  |  |     p = RepPredProp(p0 = Yap_PredPropByAtomNonThreadLocal(atom, mod)); | 
					
						
							| 
									
										
										
										
											2009-05-24 19:25:26 -05:00
										 |  |  |     if (EndOfPAEntr(p0)) { | 
					
						
							|  |  |  |       save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch, OUT_OF_HEAP_BOTCH); | 
					
						
							| 
									
										
										
										
											2009-05-24 19:25:26 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-06-22 12:40:55 -05:00
										 |  |  |     /* if we are profiling, make sure we register we entered this predicate */ | 
					
						
							|  |  |  |     if (profiling) | 
					
						
							|  |  |  |       Yap_emit(enter_profiling_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							|  |  |  |     if (call_counting) | 
					
						
							|  |  |  |       Yap_emit(count_call_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     f = FunctorOfTerm(Goal); | 
					
						
							| 
									
										
										
										
											2004-02-11 01:20:56 +00:00
										 |  |  |     p = RepPredProp(p0 = Yap_PredPropByFunctorNonThreadLocal(f, mod)); | 
					
						
							| 
									
										
										
										
											2009-05-24 19:25:26 -05:00
										 |  |  |     if (EndOfPAEntr(p0)) { | 
					
						
							|  |  |  |       save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |       siglongjmp(cglobs->cint.CompilerBotch, OUT_OF_HEAP_BOTCH); | 
					
						
							| 
									
										
										
										
											2009-05-24 19:25:26 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-08-01 13:14:17 +00:00
										 |  |  |     if (f == FunctorOr || f == FunctorVBar) { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       Term arg; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       CELL l = ++cglobs->labelno; | 
					
						
							|  |  |  |       CELL m = ++cglobs->labelno; | 
					
						
							|  |  |  |       int save = cglobs->onlast; | 
					
						
							|  |  |  |       int savegoalno = cglobs->goalno; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       int frst = TRUE; | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  |       int commitflag = 0; | 
					
						
							|  |  |  |       int looking_at_commit = FALSE; | 
					
						
							|  |  |  |       int optimizing_commit = FALSE; | 
					
						
							|  |  |  |       Term commitvar = 0; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       PInstr *FirstP = cglobs->cint.cpc, *savecpc, *savencpc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       push_branch(cglobs->onbranch, TermNil, cglobs); | 
					
						
							|  |  |  |       ++cglobs->curbranch; | 
					
						
							|  |  |  |       cglobs->onbranch = cglobs->curbranch; | 
					
						
							|  |  |  |       cglobs->or_found = TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       do { | 
					
						
							|  |  |  | 	arg = ArgOfTerm(1, Goal); | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	looking_at_commit = IsApplTerm(arg) && | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  | 		FunctorOfTerm(arg) == FunctorArrow; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	if (frst) { | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	  if (optimizing_commit) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    Yap_emit(label_op, l, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	    l = ++cglobs->labelno; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit_3ops(push_or_op, l, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	  if (looking_at_commit && | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	      Yap_is_a_test_pred(ArgOfTerm(1, arg), mod)) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    /*
 | 
					
						
							|  |  |  | 	     * let them think they are still the | 
					
						
							|  |  |  | 	     * first  | 
					
						
							|  |  |  | 	     */ | 
					
						
							| 
									
										
										
										
											2008-12-09 14:23:19 +00:00
										 |  |  | 	    //	    Yap_emit(commit_opt_op, l, Zero, &cglobs->cint);
 | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	    optimizing_commit = TRUE; | 
					
						
							| 
									
										
										
										
											2008-12-09 14:23:19 +00:00
										 |  |  | 	    Yap_emit_3ops(label_ctl_op, SPECIAL_LABEL_INIT, SPECIAL_LABEL_FAILURE, l, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	  else { | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	    optimizing_commit = FALSE; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	    cglobs->needs_env = TRUE; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    Yap_emit_3ops(either_op, l,  Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	    Yap_emit(restore_tmps_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    frst = FALSE; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	  optimizing_commit = FALSE; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit(label_op, l, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  Yap_emit(pushpop_or_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  Yap_emit_3ops(orelse_op, l = ++cglobs->labelno, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	    cglobs->needs_env = TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * if(IsApplTerm(arg) && | 
					
						
							|  |  |  | 	 * FunctorOfTerm(arg)==FunctorArrow) {  | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	if (looking_at_commit) { | 
					
						
							|  |  |  | 	  if (!optimizing_commit && !commitflag) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | 	    CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    /* This instruction is placed before
 | 
					
						
							|  |  |  | 	     * the disjunction. This means that | 
					
						
							|  |  |  | 	     * the program counter must point | 
					
						
							|  |  |  | 	     * correctly, and also that the age | 
					
						
							|  |  |  | 	     * of variable is older than the | 
					
						
							|  |  |  | 	     * current branch. | 
					
						
							|  |  |  | 	     */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    int my_goalno = cglobs->goalno; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    cglobs->goalno = savegoalno; | 
					
						
							|  |  |  | 	    commitflag = cglobs->labelno; | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	    commitvar = MkVarTerm(); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	    if (HR == (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	      /* oops, too many new variables */ | 
					
						
							|  |  |  | 	      save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	      siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    savecpc = cglobs->cint.cpc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    savencpc = FirstP->nextInst; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    cglobs->cint.cpc = FirstP; | 
					
						
							|  |  |  | 	    cglobs->onbranch = pop_branch(cglobs); | 
					
						
							|  |  |  | 	    c_var(commitvar, save_b_flag, 1, 0, cglobs); | 
					
						
							|  |  |  | 	    push_branch(cglobs->onbranch,  commitvar, cglobs); | 
					
						
							|  |  |  | 	    cglobs->onbranch = cglobs->curbranch; | 
					
						
							|  |  |  | 	    cglobs->cint.cpc->nextInst = savencpc; | 
					
						
							|  |  |  | 	    cglobs->cint.cpc = savecpc; | 
					
						
							|  |  |  | 	    cglobs->goalno = my_goalno; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  save = cglobs->onlast; | 
					
						
							|  |  |  | 	  cglobs->onlast = FALSE; | 
					
						
							|  |  |  | 	  c_goal(ArgOfTerm(1, arg), mod, cglobs); | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	  if (!optimizing_commit) { | 
					
						
							|  |  |  | 	    c_var((Term) commitvar, commit_b_flag, | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 		  1, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2008-12-09 14:23:19 +00:00
										 |  |  | 	  } else { | 
					
						
							|  |  |  | 	    Yap_emit_3ops(label_ctl_op, SPECIAL_LABEL_CLEAR, SPECIAL_LABEL_FAILURE, l, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  cglobs->onlast = save; | 
					
						
							|  |  |  | 	  c_goal(ArgOfTerm(2, arg), mod, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	else { | 
					
						
							|  |  |  | 	  /* standard disjunction */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  c_goal(ArgOfTerm(1, Goal), mod, cglobs); | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (!cglobs->onlast) { | 
					
						
							|  |  |  | 	  Yap_emit(jump_op, m, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2008-12-09 14:23:19 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 	   | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	if (!optimizing_commit || !cglobs->onlast) { | 
					
						
							|  |  |  | 	  cglobs->goalno = savegoalno + 1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	Goal = ArgOfTerm(2, Goal); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	++cglobs->curbranch; | 
					
						
							|  |  |  | 	cglobs->onbranch = cglobs->curbranch; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } while (IsNonVarTerm(Goal) && IsApplTerm(Goal) | 
					
						
							| 
									
										
										
										
											2006-08-01 13:14:17 +00:00
										 |  |  | 	       && (FunctorOfTerm(Goal) == FunctorOr | 
					
						
							|  |  |  | 		   || FunctorOfTerm(Goal) == FunctorVBar)); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(pushpop_or_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(label_op, l, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       if (!optimizing_commit) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit(orlast_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  | 	optimizing_commit = FALSE;	/* not really necessary */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       c_goal(Goal, mod, cglobs); | 
					
						
							|  |  |  |       /*              --cglobs->onbranch; */ | 
					
						
							|  |  |  |       cglobs->onbranch = pop_branch(cglobs); | 
					
						
							|  |  |  |       if (!cglobs->onlast) { | 
					
						
							|  |  |  | 	Yap_emit(label_op, m, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	if ((cglobs->onlast = save)) | 
					
						
							|  |  |  | 	  c_goal(MkAtomTerm(AtomTrue), mod, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(pop_or_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (f == FunctorComma) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       int save = cglobs->onlast; | 
					
						
							| 
									
										
										
										
											2005-03-13 06:26:13 +00:00
										 |  |  |       Term t2 = ArgOfTerm(2, Goal); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs->onlast = FALSE; | 
					
						
							|  |  |  |       c_goal(ArgOfTerm(1, Goal), mod, cglobs); | 
					
						
							|  |  |  |       cglobs->onlast = save; | 
					
						
							|  |  |  |       c_goal(t2, mod, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (f == FunctorNot || f == FunctorAltNot) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |       CACHE_REGS | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       CELL label = (cglobs->labelno += 2); | 
					
						
							|  |  |  |       CELL end_label = (cglobs->labelno += 2); | 
					
						
							|  |  |  |       int save = cglobs->onlast; | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  |       Term commitvar; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       /* for now */ | 
					
						
							|  |  |  |       cglobs->needs_env = TRUE; | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  |       commitvar = MkVarTerm(); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |       if (HR == (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* oops, too many new variables */ | 
					
						
							|  |  |  | 	save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       push_branch(cglobs->onbranch, commitvar, cglobs); | 
					
						
							|  |  |  |       ++cglobs->curbranch; | 
					
						
							|  |  |  |       cglobs->onbranch = cglobs->curbranch; | 
					
						
							|  |  |  |       cglobs->or_found = TRUE; | 
					
						
							|  |  |  |       cglobs->onlast = FALSE; | 
					
						
							|  |  |  |       c_var(commitvar, save_b_flag, 1, 0, cglobs); | 
					
						
							|  |  |  |       Yap_emit_3ops(push_or_op, label, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit_3ops(either_op, label,  Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(restore_tmps_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       c_goal(ArgOfTerm(1, Goal), mod, cglobs); | 
					
						
							|  |  |  |       c_var(commitvar, commit_b_flag, 1, 0, cglobs); | 
					
						
							|  |  |  |       cglobs->onlast = save; | 
					
						
							|  |  |  |       Yap_emit(fail_op, end_label, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(pushpop_or_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(label_op, label, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(orlast_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       Yap_emit(label_op, end_label, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       cglobs->onlast = save; | 
					
						
							|  |  |  |       /*              --cglobs->onbranch; */ | 
					
						
							|  |  |  |       cglobs->onbranch = pop_branch(cglobs); | 
					
						
							|  |  |  |       c_goal(MkAtomTerm(AtomTrue), mod, cglobs); | 
					
						
							|  |  |  |       ++cglobs->goalno; | 
					
						
							|  |  |  |       Yap_emit(pop_or_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (f == FunctorArrow) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |       CACHE_REGS | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  |       Term commitvar; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       int save = cglobs->onlast; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  |       commitvar = MkVarTerm(); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |       if (HR == (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	/* oops, too many new variables */ | 
					
						
							|  |  |  | 	save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs->onlast = FALSE; | 
					
						
							|  |  |  |       c_var(commitvar, save_b_flag, 1, 0, cglobs); | 
					
						
							|  |  |  |       c_goal(ArgOfTerm(1, Goal), mod, cglobs); | 
					
						
							|  |  |  |       c_var(commitvar, commit_b_flag, 1, 0, cglobs); | 
					
						
							|  |  |  |       cglobs->onlast = save; | 
					
						
							|  |  |  |       c_goal(ArgOfTerm(2, Goal), mod, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       return; | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-22 23:28:04 -05:00
										 |  |  |     else if (f == FunctorEq) { | 
					
						
							| 
									
										
										
										
											2009-06-22 12:40:55 -05:00
										 |  |  |       if (profiling) | 
					
						
							|  |  |  | 	Yap_emit(enter_profiling_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       else if (call_counting) | 
					
						
							|  |  |  | 	Yap_emit(count_call_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       c_eq(ArgOfTerm(1, Goal), ArgOfTerm(2, Goal), cglobs); | 
					
						
							|  |  |  |       if (cglobs->onlast) { | 
					
						
							|  |  |  | 	Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	PELOCK(44,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (is_tabled(cglobs->cint.CurrentPred)) | 
					
						
							|  |  |  | 	  Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return; | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-03-10 16:24:26 +00:00
										 |  |  |     else if (f == FunctorSafe) { | 
					
						
							|  |  |  |       Ventry *v = (Ventry *)ArgOfTerm(1, Goal); | 
					
						
							|  |  |  |       /* This variable must be known before */ | 
					
						
							|  |  |  |       v->FlagsOfVE |= SafeVar; | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-22 23:28:04 -05:00
										 |  |  |     else if (p->PredFlags & AsmPredFlag) { | 
					
						
							| 
									
										
										
										
											2010-02-22 23:08:40 +00:00
										 |  |  |       basic_preds op = p->PredFlags & 0x7f; | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-22 12:40:55 -05:00
										 |  |  |       if (profiling) | 
					
						
							|  |  |  | 	Yap_emit(enter_profiling_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       else if (call_counting) | 
					
						
							|  |  |  | 	Yap_emit(count_call_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       if (op >= _atom && op <= _primitive) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_test(op, ArgOfTerm(1, Goal), cglobs); | 
					
						
							|  |  |  | 	if (cglobs->onlast) { | 
					
						
							|  |  |  | 	  Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	  PELOCK(45,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  if (is_tabled(cglobs->cint.CurrentPred)) | 
					
						
							|  |  |  | 	    Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	  UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return; | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       else if (op >= _plus && op <= _functor) { | 
					
						
							| 
									
										
										
										
											2009-06-22 16:03:14 -05:00
										 |  |  | 	if (profiling) | 
					
						
							| 
									
										
										
										
											2009-06-25 22:31:02 -05:00
										 |  |  | 	  Yap_emit(enter_profiling_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2009-06-22 16:03:14 -05:00
										 |  |  | 	else if (call_counting) | 
					
						
							| 
									
										
										
										
											2009-06-25 22:31:02 -05:00
										 |  |  | 	  Yap_emit(count_call_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	if (op == _functor) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  c_functor(Goal, mod, cglobs); | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	  c_bifun(op, | 
					
						
							|  |  |  | 		  ArgOfTerm(1, Goal), | 
					
						
							|  |  |  | 		  ArgOfTerm(2, Goal), | 
					
						
							| 
									
										
										
										
											2001-11-15 00:01:43 +00:00
										 |  |  | 		  ArgOfTerm(3, Goal), | 
					
						
							| 
									
										
										
										
											2007-03-27 13:48:51 +00:00
										 |  |  | 		  Goal, | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 		  mod, | 
					
						
							|  |  |  | 		  cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-23 20:41:58 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (cglobs->onlast) { | 
					
						
							|  |  |  | 	  Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	  PELOCK(46,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  if (is_tabled(cglobs->cint.CurrentPred)) | 
					
						
							|  |  |  | 	    Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	    Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	  UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return; | 
					
						
							| 
									
										
										
										
											2009-03-09 11:52:01 +00:00
										 |  |  |       } else if (op == _p_label_ctl) { | 
					
						
							|  |  |  | 	emit_special_label(Goal, cglobs); | 
					
						
							|  |  |  | 	return; | 
					
						
							| 
									
										
										
										
											2008-12-09 12:54:27 +00:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_args(Goal, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							| 
									
										
										
										
											2009-02-09 21:56:40 +00:00
										 |  |  |     else if (p->PredFlags & BinaryPredFlag && !EAM) { | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2009-07-22 23:28:04 -05:00
										 |  |  |     else if (p->PredFlags & BinaryPredFlag) { | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |       CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       Term a1 = ArgOfTerm(1,Goal); | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       if (IsVarTerm(a1) && !IsNewVar(a1)) { | 
					
						
							|  |  |  | 	Term a2 = ArgOfTerm(2,Goal); | 
					
						
							|  |  |  | 	if (IsVarTerm(a2) && !IsNewVar(a2)) { | 
					
						
							|  |  |  | 	  if (IsNewVar(a2))  { | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	    LOCAL_Error_TYPE = INSTANTIATION_ERROR; | 
					
						
							|  |  |  | 	    LOCAL_Error_Term = a2; | 
					
						
							|  |  |  | 	    LOCAL_ErrorMessage = LOCAL_ErrorSay; | 
					
						
							|  |  |  | 	    sprintf(LOCAL_ErrorMessage, "compiling %s/2 with second arg unbound",  RepAtom(NameOfFunctor(p->FunctorOfPred))->StrOfAE); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	    siglongjmp(cglobs->cint.CompilerBotch,1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  c_var(a1, bt1_flag, 2, 0, cglobs); | 
					
						
							|  |  |  | 	  cglobs->current_p0 = p0; | 
					
						
							|  |  |  | 	  c_var(a2, bt2_flag, 2, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  Term t2 = MkVarTerm(); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	  if (HR == (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    /* oops, too many new variables */ | 
					
						
							|  |  |  | 	    save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	    siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  c_eq(t2, a2, cglobs); | 
					
						
							|  |  |  | 	  c_var(a1, bt1_flag, 2, 0, cglobs); | 
					
						
							|  |  |  | 	  cglobs->current_p0 = p0; | 
					
						
							|  |  |  | 	  c_var(t2, bt2_flag, 2, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	Term a2 = ArgOfTerm(2,Goal); | 
					
						
							|  |  |  | 	Term t1 = MkVarTerm(); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	if (HR == (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  /* oops, too many new variables */ | 
					
						
							|  |  |  | 	  save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	  siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_eq(t1, a1, cglobs); | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	if (IsVarTerm(a2) && !IsNewVar(a2)) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  c_var(t1, bt1_flag, 2, 0, cglobs); | 
					
						
							|  |  |  | 	  cglobs->current_p0 = p0; | 
					
						
							|  |  |  | 	  c_var(a2, bt2_flag, 2, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  Term t2 = MkVarTerm(); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	  if (HR == (CELL *)cglobs->cint.freep0) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    /* oops, too many new variables */ | 
					
						
							|  |  |  | 	    save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	    siglongjmp(cglobs->cint.CompilerBotch,OUT_OF_TEMPS_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  c_eq(t2, a2, cglobs); | 
					
						
							|  |  |  | 	  c_var(t1, bt1_flag, 2, 0, cglobs); | 
					
						
							|  |  |  | 	  cglobs->current_p0 = p0; | 
					
						
							|  |  |  | 	  c_var(t2, bt2_flag, 2, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       if (cglobs->onlast) { | 
					
						
							|  |  |  | 	Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	PELOCK(47,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (is_tabled(cglobs->cint.CurrentPred)) | 
					
						
							|  |  |  | 	  Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return; | 
					
						
							| 
									
										
										
										
											2003-01-29 14:47:17 +00:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2009-06-22 12:40:55 -05:00
										 |  |  |       if (profiling) | 
					
						
							|  |  |  | 	Yap_emit(enter_profiling_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       else if (call_counting) | 
					
						
							|  |  |  | 	Yap_emit(count_call_op, (CELL)p, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2003-01-29 14:47:17 +00:00
										 |  |  |       if (f == FunctorExecuteInMod) { | 
					
						
							|  |  |  | 	/* compile the first argument only */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_arg(1, ArgOfTerm(1,Goal), 1, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2003-01-29 14:47:17 +00:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	c_args(Goal, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2003-01-29 14:47:17 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (p->PredFlags & SafePredFlag | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef YAPOR
 | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |       /* synchronisation means saving the state, so it is never safe in YAPOR */ | 
					
						
							|  |  |  |       && !(p->PredFlags & SyncPredFlag) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* YAPOR */
 | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |       ) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_emit(safe_call_op, (CELL) p0, Zero, &cglobs->cint); | 
					
						
							|  |  |  |     if (cglobs->onlast) { | 
					
						
							|  |  |  |       Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  |       PELOCK(48,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       if (is_tabled(cglobs->cint.CurrentPred)) | 
					
						
							|  |  |  | 	Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  |       UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2008-08-28 04:43:00 +01:00
										 |  |  |     if ((p->PredFlags & (AsmPredFlag |  | 
					
						
							|  |  |  | 			ModuleTransparentPredFlag | | 
					
						
							|  |  |  | 			 UserCPredFlag)) || | 
					
						
							|  |  |  | 	p->FunctorOfPred == FunctorExecuteInMod) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef YAPOR
 | 
					
						
							|  |  |  |       if (p->PredFlags & SyncPredFlag) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit(sync_op, (CELL)p, (CELL)(p->ArityOfPE), &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* YAPOR */
 | 
					
						
							| 
									
										
										
										
											2003-01-29 14:47:17 +00:00
										 |  |  |       if (p->FunctorOfPred == FunctorExecuteInMod) { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	cglobs->needs_env = TRUE; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit_4ops(call_op, (CELL) p0, Zero, Zero, ArgOfTerm(2,Goal), &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2003-01-29 14:47:17 +00:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	cglobs->needs_env = TRUE; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit_3ops(call_op, (CELL) p0, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2003-01-29 14:47:17 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* functor is allowed to call the garbage collector */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       if (cglobs->onlast) { | 
					
						
							|  |  |  | 	Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	cglobs->or_found = TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	PELOCK(49,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (is_tabled(cglobs->cint.CurrentPred)) | 
					
						
							|  |  |  | 	  Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       if (cglobs->onlast) { | 
					
						
							|  |  |  | 	Yap_emit(deallocate_op, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	PELOCK(50,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (is_tabled(cglobs->cint.CurrentPred)) { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	  cglobs->needs_env = TRUE; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit_3ops(call_op, (CELL) p0, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	  Yap_emit(table_new_answer_op, Zero, cglobs->cint.CurrentPred->ArityOfPE, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  Yap_emit(execute_op, (CELL) p0, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-10-03 13:54:35 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	cglobs->needs_env = TRUE; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit_3ops(call_op, (CELL) p0, Zero, Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     if (!cglobs->onlast) | 
					
						
							|  |  |  |       ++cglobs->goalno; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2009-04-07 15:44:46 +01:00
										 |  |  | c_body(Term Body, Term mod, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->onhead = FALSE; | 
					
						
							|  |  |  |   cglobs->BodyStart = cglobs->cint.cpc; | 
					
						
							|  |  |  |   cglobs->goalno = 1; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   while (IsNonVarTerm(Body) && IsApplTerm(Body) | 
					
						
							|  |  |  | 	 && FunctorOfTerm(Body) == FunctorComma) { | 
					
						
							| 
									
										
										
										
											2001-05-07 19:56:02 +00:00
										 |  |  |     Term t2 = ArgOfTerm(2, Body); | 
					
						
							| 
									
										
										
										
											2009-02-09 21:56:40 +00:00
										 |  |  |     if (!cglobs->cint.success_handler && IsTrueGoal(t2)) { | 
					
						
							| 
									
										
										
										
											2001-05-07 19:56:02 +00:00
										 |  |  |       /* optimise the case where some idiot left trues at the end
 | 
					
						
							|  |  |  | 	 of the clause. | 
					
						
							|  |  |  |       */ | 
					
						
							|  |  |  |       Body = ArgOfTerm(1, Body); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     c_goal(ArgOfTerm(1, Body), mod, cglobs); | 
					
						
							| 
									
										
										
										
											2001-05-07 19:56:02 +00:00
										 |  |  |     Body = t2; | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  |     if (EAM) Yap_emit(endgoal_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->onlast = TRUE; | 
					
						
							|  |  |  |   c_goal(Body, mod, cglobs); | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  |     if (EAM && cglobs->goalno > 1) { | 
					
						
							|  |  |  |       if (cglobs->cint.cpc->op==procceed_op) { | 
					
						
							|  |  |  |   	cglobs->cint.cpc->op=endgoal_op; | 
					
						
							|  |  |  |         Yap_emit(procceed_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       } else | 
					
						
							|  |  |  |         Yap_emit(endgoal_op, Zero, Zero, &cglobs->cint); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | c_head(Term t, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   Functor f; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->goalno = 0; | 
					
						
							|  |  |  |   cglobs->onhead = TRUE; | 
					
						
							|  |  |  |   cglobs->onlast = FALSE; | 
					
						
							|  |  |  |   cglobs->curbranch = cglobs->onbranch = 0; | 
					
						
							|  |  |  |   cglobs->branch_pointer = cglobs->parent_branches; | 
					
						
							| 
									
										
										
										
											2010-08-02 13:04:30 +01:00
										 |  |  |   cglobs->space_used = 0; | 
					
						
							|  |  |  |   cglobs->space_op = NULL; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (IsAtomTerm(t)) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_emit(name_op, (CELL) AtomOfTerm(t), Zero, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							| 
									
										
										
										
											2013-01-15 22:58:34 +00:00
										 |  |  |     if (EAM) { | 
					
						
							|  |  |  |       Yap_emit(run_op,Zero,(UInt) cglobs->cint.CurrentPred,&cglobs->cint); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-01-15 22:58:34 +00:00
										 |  |  |     Yap_emit(ensure_space_op, Zero , Zero, &cglobs->cint); | 
					
						
							|  |  |  |     cglobs->space_op = cglobs->cint.cpc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   f = FunctorOfTerm(t); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   Yap_emit(name_op, (CELL) NameOfFunctor(f), ArityOfFunctor(f), &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  |    if (EAM) { | 
					
						
							| 
									
										
										
										
											2010-05-10 22:14:08 +01:00
										 |  |  |      Yap_emit(run_op,Zero,(UInt) cglobs->cint.CurrentPred,&cglobs->cint); | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  |    } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-01-15 22:58:34 +00:00
										 |  |  |   if (Yap_ExecutionMode == MIXED_MODE_USER) | 
					
						
							| 
									
										
										
										
											2009-04-08 00:32:36 +01:00
										 |  |  |      Yap_emit(native_op, 0, 0, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2013-01-15 22:58:34 +00:00
										 |  |  |   Yap_emit(ensure_space_op, Zero , Zero, &cglobs->cint); | 
					
						
							|  |  |  |   cglobs->space_op = cglobs->cint.cpc; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   c_args(t, 0, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | inline static int | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | usesvar(compiler_vm_op ic) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   if (ic >= get_var_op && ic <= put_val_op) | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     return TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   switch (ic) { | 
					
						
							|  |  |  |   case save_b_op: | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  |   case commit_b_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   case patch_b_op: | 
					
						
							|  |  |  |   case save_appl_op: | 
					
						
							|  |  |  |   case save_pair_op: | 
					
						
							|  |  |  |   case f_val_op: | 
					
						
							|  |  |  |   case f_var_op: | 
					
						
							|  |  |  |   case fetch_args_for_bccall: | 
					
						
							|  |  |  |   case bccall_op: | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  |     return TRUE; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   default: | 
					
						
							|  |  |  |     break; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  |   if (ic >= unify_s_var_op && ic <= write_s_val_op) | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  |     return TRUE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   return ((ic >= unify_var_op && ic <= write_val_op) | 
					
						
							|  |  |  | 	  || | 
					
						
							|  |  |  | 	  (ic >= unify_last_var_op && ic <= unify_last_val_op)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Do as in the traditional WAM and make sure voids are in | 
					
						
							|  |  |  |  * environments | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define LOCALISE_VOIDS 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef LOCALISE_VOIDS
 | 
					
						
							|  |  |  | typedef  struct env_tmp { | 
					
						
							|  |  |  |   Ventry * Var; | 
					
						
							|  |  |  |   struct env_tmp *Next; | 
					
						
							|  |  |  | }  EnvTmp; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | AssignPerm(PInstr *pc, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-07 15:18:43 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   int uses_var; | 
					
						
							|  |  |  |   PInstr *opc = NULL; | 
					
						
							|  |  |  | #ifdef LOCALISE_VOIDS
 | 
					
						
							|  |  |  |   EnvTmp *EnvTmps = NULL; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* The WAM tries to keep voids on the
 | 
					
						
							|  |  |  |    * environment. Traditionally, YAP liberally globalises | 
					
						
							|  |  |  |    * voids. | 
					
						
							|  |  |  |    *  | 
					
						
							|  |  |  |    * The new version goes to some length to keep void variables | 
					
						
							|  |  |  |    * in environments, but it is dubious that improves | 
					
						
							|  |  |  |    * performance, and may actually slow down the system | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   while (pc != NULL) { | 
					
						
							|  |  |  |     PInstr *tpc = pc->nextInst; | 
					
						
							|  |  |  | #ifdef LOCALISE_VOIDS
 | 
					
						
							|  |  |  |     if (pc->op == put_var_op) { | 
					
						
							|  |  |  |       Ventry *v = (Ventry *) (pc->rnd1); | 
					
						
							|  |  |  |       if (v->AgeOfVE == v->FirstOfVE | 
					
						
							|  |  |  | 	  && !(v->FlagsOfVE & (GlobalVal|OnHeadFlag|OnLastGoal|NonVoid)) ) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	EnvTmp *x = (EnvTmp *)Yap_AllocCMem(sizeof(*x), &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	x->Next = EnvTmps; | 
					
						
							|  |  |  | 	x->Var = v; | 
					
						
							|  |  |  | 	EnvTmps = x; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     } else | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     if (pc->op == call_op || pc->op == either_op || pc->op == orelse_op || pc->op == push_or_op) { | 
					
						
							|  |  |  | #ifdef LOCALISE_VOIDS
 | 
					
						
							|  |  |  |       pc->ops.opseqt[1] = (CELL)EnvTmps; | 
					
						
							|  |  |  |       if (EnvTmps) | 
					
						
							|  |  |  | 	EnvTmps = NULL; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pc->nextInst = opc; | 
					
						
							|  |  |  |     opc = pc; | 
					
						
							|  |  |  |     pc = tpc; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   pc = opc; | 
					
						
							|  |  |  |   opc = NULL; | 
					
						
							|  |  |  |   do { | 
					
						
							|  |  |  |     PInstr *npc = pc->nextInst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pc->nextInst = opc; | 
					
						
							|  |  |  |     uses_var = usesvar(pc->op); | 
					
						
							|  |  |  |     if (uses_var) { | 
					
						
							|  |  |  |       Ventry *v = (Ventry *) (pc->rnd1); | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  |    if (EAM) { | 
					
						
							|  |  |  |       if (v->NoOfVE == Unassigned || v->KindOfVE!=PermVar) { | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  | 	v->NoOfVE = PermVar | (LOCAL_nperm++); | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 	v->KindOfVE = PermVar; | 
					
						
							|  |  |  | 	v->FlagsOfVE |= PermFlag;	 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |    } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       if (v->NoOfVE == Unassigned) { | 
					
						
							|  |  |  | 	if ((v->AgeOfVE > 1 && (v->AgeOfVE > v->FirstOfVE)) | 
					
						
							|  |  |  | 	    || v->KindOfVE == PermVar	/*
 | 
					
						
							|  |  |  | 					 * * || (v->FlagsOfVE & NonVoid && !(v->FlagsOfVE & | 
					
						
							|  |  |  | 					 * * OnHeadFlag))  | 
					
						
							|  |  |  | 					 */ ) { | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  | 	  v->NoOfVE = PermVar | (LOCAL_nperm++); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  v->KindOfVE = PermVar; | 
					
						
							|  |  |  | 	  v->FlagsOfVE |= PermFlag; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  v->NoOfVE = v->KindOfVE = TempVar; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-07-05 16:14:15 +00:00
										 |  |  |     } else if (pc->op == empty_call_op) { | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  |       pc->rnd2 = LOCAL_nperm; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } else if (pc->op == call_op || pc->op == either_op || pc->op == orelse_op || pc->op == push_or_op) { | 
					
						
							|  |  |  | #ifdef LOCALISE_VOIDS
 | 
					
						
							|  |  |  |       EnvTmps = (EnvTmp *)(pc->ops.opseqt[1]); | 
					
						
							|  |  |  |       while (EnvTmps) { | 
					
						
							|  |  |  | 	Ventry *v = EnvTmps->Var; | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  | 	v->NoOfVE = PermVar | (LOCAL_nperm++); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	v->KindOfVE = PermVar; | 
					
						
							|  |  |  | 	v->FlagsOfVE |= (PermFlag|SafeVar); | 
					
						
							|  |  |  | 	EnvTmps = EnvTmps->Next; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  |       pc->rnd2 = LOCAL_nperm; | 
					
						
							| 
									
										
										
										
											2011-02-14 11:29:20 -08:00
										 |  |  |     } else if (pc->op == cut_op || | 
					
						
							|  |  |  | 	       pc->op == cutexit_op || | 
					
						
							|  |  |  | 	       pc->op == commit_b_op) { | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  |       pc->rnd2 = LOCAL_nperm; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     opc = pc; | 
					
						
							|  |  |  |     pc = npc; | 
					
						
							|  |  |  |   } while (pc != NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static CELL * | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | init_bvarray(int nperm, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   CELL *vinfo = NULL; | 
					
						
							| 
									
										
										
										
											2010-05-11 00:18:12 +01:00
										 |  |  |   size_t sz = sizeof(CELL)*(1+nperm/(8*sizeof(CELL))); | 
					
						
							|  |  |  |   vinfo = (CELL *)Yap_AllocCMem(sz, &cglobs->cint); | 
					
						
							|  |  |  |   memset((void *)vinfo, 0, sz); | 
					
						
							|  |  |  |   return vinfo; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | clear_bvarray(int var, CELL *bvarray | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  | 	      , compiler_struct *cglobs | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   int max = 8*sizeof(CELL); | 
					
						
							|  |  |  |   CELL nbit; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* get to the array position */ | 
					
						
							|  |  |  |   while (var >= max) { | 
					
						
							|  |  |  |     bvarray++; | 
					
						
							|  |  |  |     var -= max; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* now put a 0 on it, from now on the variable is initialised */ | 
					
						
							| 
									
										
										
										
											2010-05-11 00:18:12 +01:00
										 |  |  |   nbit = ((CELL)1 << var); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   if (*bvarray & nbit) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |     CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* someone had already marked this variable: complain */ | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_TYPE = INTERNAL_COMPILER_ERROR; | 
					
						
							|  |  |  |     LOCAL_Error_Term = TermNil; | 
					
						
							|  |  |  |     LOCAL_ErrorMessage = "compiler internal error: variable initialised twice"; | 
					
						
							| 
									
										
										
										
											2010-05-11 00:18:12 +01:00
										 |  |  |  fprintf(stderr," vsc: compiling7\n"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |     siglongjmp(cglobs->cint.CompilerBotch, COMPILER_ERR_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->pbvars++; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   *bvarray |= nbit; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* copy the current state of the perm variable state array to code space */ | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | add_bvarray_op(PInstr *cp, CELL *bvarray, int env_size, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   int i, size = env_size/(8*sizeof(CELL)); | 
					
						
							|  |  |  |   CELL *dest; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dest =  | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_emit_extra_size(mark_initialised_pvars_op, (CELL)env_size, (size+1)*sizeof(CELL), &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* copy the cells to dest */ | 
					
						
							|  |  |  |   for (i = 0; i <= size; i++) | 
					
						
							|  |  |  |     *dest++ = *bvarray++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* vsc: this code is not working, as it is too complex */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |   int lab; | 
					
						
							|  |  |  |   int last; | 
					
						
							|  |  |  |   PInstr *pc; | 
					
						
							|  |  |  | }  bventry; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-11 15:08:03 +00:00
										 |  |  | #define MAX_DISJUNCTIONS 128
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | static bventry *bvstack; | 
					
						
							|  |  |  | static int bvindex = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | push_bvmap(int label, PInstr *pcpc, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   if (bvindex == MAX_DISJUNCTIONS) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |     CACHE_REGS | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_TYPE = INTERNAL_COMPILER_ERROR; | 
					
						
							|  |  |  |     LOCAL_Error_Term = TermNil; | 
					
						
							|  |  |  |     LOCAL_ErrorMessage = "Too many embedded disjunctions"; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |     siglongjmp(cglobs->cint.CompilerBotch, COMPILER_ERR_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* the label instruction */ | 
					
						
							|  |  |  |   bvstack[bvindex].lab = label; | 
					
						
							|  |  |  |   bvstack[bvindex].last = -1; | 
					
						
							|  |  |  |   /* where we have the code */ | 
					
						
							| 
									
										
										
										
											2002-11-11 17:38:10 +00:00
										 |  |  |   bvstack[bvindex].pc = pcpc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   bvindex++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | reset_bvmap(CELL *bvarray, int nperm, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   int size, size1, env_size, i; | 
					
						
							|  |  |  |   CELL *source; | 
					
						
							| 
									
										
										
										
											2005-02-21 16:50:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (bvarray == NULL) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (bvindex == 0) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |     CACHE_REGS | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_TYPE = INTERNAL_COMPILER_ERROR; | 
					
						
							|  |  |  |     LOCAL_Error_Term = TermNil; | 
					
						
							|  |  |  |     LOCAL_ErrorMessage = "No embedding in disjunctions"; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |     siglongjmp(cglobs->cint.CompilerBotch, COMPILER_ERR_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   env_size = (bvstack[bvindex-1].pc)->rnd1; | 
					
						
							|  |  |  |   size = env_size/(8*sizeof(CELL)); | 
					
						
							|  |  |  |   size1 = nperm/(8*sizeof(CELL)); | 
					
						
							|  |  |  |   source = (bvstack[bvindex-1].pc)->arnds; | 
					
						
							|  |  |  |   for (i = 0; i <= size; i++) | 
					
						
							|  |  |  |     *bvarray++ = *source++; | 
					
						
							|  |  |  |   for (i = size+1; i<= size1; i++) | 
					
						
							|  |  |  |     *bvarray++ = (CELL)(0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | pop_bvmap(CELL *bvarray, int nperm, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   if (bvindex == 0) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |     CACHE_REGS | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_TYPE = INTERNAL_COMPILER_ERROR; | 
					
						
							|  |  |  |     LOCAL_Error_Term = TermNil; | 
					
						
							|  |  |  |     LOCAL_ErrorMessage = "Too few embedded disjunctions"; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /*  save_machine_regs();
 | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch, OUT_OF_HEAP_BOTCH); */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   reset_bvmap(bvarray, nperm, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   bvindex--; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |   PInstr *p; | 
					
						
							|  |  |  |   Ventry *v; | 
					
						
							|  |  |  | } UnsafeEntry; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* extend to also support variable usage bitmaps for garbage collection */ | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | CheckUnsafe(PInstr *pc, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-07 15:18:43 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   int pending = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* say that all variables are yet to initialise */ | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  |   CELL *vstat = init_bvarray(LOCAL_nperm, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   UnsafeEntry *UnsafeStack = | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  |     (UnsafeEntry *) Yap_AllocCMem(LOCAL_nperm * sizeof(UnsafeEntry), &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   /* keep a copy of previous cglobs->cint.cpc and CodeStart */ | 
					
						
							|  |  |  |   PInstr *opc = cglobs->cint.cpc; | 
					
						
							|  |  |  |   PInstr *OldCodeStart = cglobs->cint.CodeStart; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->cint.CodeStart = cglobs->cint.BlobsStart; | 
					
						
							|  |  |  |   cglobs->cint.cpc = cglobs->cint.icpc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   bvindex = 0; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   bvstack = (bventry *)Yap_AllocCMem(MAX_DISJUNCTIONS * sizeof(bventry), &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   while (pc != NIL) { | 
					
						
							|  |  |  |     switch(pc->op) { | 
					
						
							|  |  |  |     case put_val_op: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Ventry *v = (Ventry *) (pc->rnd1); | 
					
						
							|  |  |  | 	if ((v->FlagsOfVE & PermFlag) && !(v->FlagsOfVE & SafeVar)) { | 
					
						
							|  |  |  | 	  UnsafeStack[pending].p = pc; | 
					
						
							|  |  |  | 	  UnsafeStack[pending++].v = v; | 
					
						
							|  |  |  | 	  v->FlagsOfVE |= SafeVar; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case put_var_op: | 
					
						
							|  |  |  |     case get_var_op: | 
					
						
							|  |  |  |     case save_b_op: | 
					
						
							|  |  |  |     case unify_var_op: | 
					
						
							|  |  |  |     case unify_last_var_op: | 
					
						
							|  |  |  |     case write_var_op: | 
					
						
							|  |  |  |     case save_appl_op: | 
					
						
							|  |  |  |     case save_pair_op: | 
					
						
							|  |  |  |     case f_var_op: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Ventry *v = (Ventry *) (pc->rnd1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (v->FlagsOfVE & PermFlag && pc == v->FirstOpForV) { | 
					
						
							|  |  |  | 	  /* the second condition covers cases such as save_b_op
 | 
					
						
							|  |  |  | 	     in a disjunction */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  clear_bvarray((v->NoOfVE & MaskVarAdrs), vstat | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  | 			, cglobs | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 			); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case push_or_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(label_op, ++cglobs->labelno, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       pc->ops.opseqt[1] = (CELL)cglobs->labelno; | 
					
						
							|  |  |  |       add_bvarray_op(pc, vstat, pc->rnd2, cglobs); | 
					
						
							|  |  |  |       push_bvmap((CELL)cglobs->labelno, cglobs->cint.cpc, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case either_op: | 
					
						
							|  |  |  |       /* add a first entry to the array */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(label_op, ++cglobs->labelno, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       pc->ops.opseqt[1] = (CELL)cglobs->labelno; | 
					
						
							|  |  |  |       add_bvarray_op(pc, vstat, pc->rnd2, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case pushpop_or_op: | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  |       reset_bvmap(vstat, LOCAL_nperm, cglobs); | 
					
						
							| 
									
										
										
										
											2006-04-12 20:08:51 +00:00
										 |  |  |       goto reset_safe_map; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case orelse_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(label_op, ++cglobs->labelno, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       pc->ops.opseqt[1] = (CELL)cglobs->labelno; | 
					
						
							|  |  |  |       add_bvarray_op(pc, vstat, pc->rnd2, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case pop_or_op: | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  |       pop_bvmap(vstat, LOCAL_nperm, cglobs); | 
					
						
							| 
									
										
										
										
											2006-04-12 20:08:51 +00:00
										 |  |  |       goto reset_safe_map; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case empty_call_op: | 
					
						
							|  |  |  |       /* just get ourselves a label describing how
 | 
					
						
							|  |  |  | 	 many permanent variables are alive */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(label_op, ++cglobs->labelno, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       pc->rnd1 = (CELL)cglobs->labelno; | 
					
						
							|  |  |  |       add_bvarray_op(pc, vstat, pc->rnd2, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2003-10-17 02:11:21 +00:00
										 |  |  |     case cut_op: | 
					
						
							|  |  |  |     case cutexit_op: | 
					
						
							|  |  |  |       /* just get ourselves a label describing how
 | 
					
						
							|  |  |  | 	 many permanent variables are alive */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(label_op, ++cglobs->labelno, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       pc->rnd1 = (CELL)cglobs->labelno; | 
					
						
							|  |  |  |       add_bvarray_op(pc, vstat, pc->rnd2, cglobs); | 
					
						
							| 
									
										
										
										
											2003-10-17 02:11:21 +00:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case call_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       Yap_emit(label_op, ++cglobs->labelno, Zero, &cglobs->cint); | 
					
						
							|  |  |  |       pc->ops.opseqt[1] = (CELL)cglobs->labelno; | 
					
						
							|  |  |  |       add_bvarray_op(pc, vstat, pc->rnd2, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case deallocate_op: | 
					
						
							| 
									
										
										
										
											2006-04-12 20:08:51 +00:00
										 |  |  |     reset_safe_map: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	int n = pc->op == call_op ? pc->rnd2 : 0; | 
					
						
							|  |  |  | 	int no; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (pending) { | 
					
						
							|  |  |  | 	  Ventry *v = UnsafeStack[--pending].v; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  v->FlagsOfVE &= ~SafeVar; | 
					
						
							|  |  |  | 	  no = (v->NoOfVE) & MaskVarAdrs; | 
					
						
							|  |  |  | 	  if (no >= n) | 
					
						
							|  |  |  | 	    UnsafeStack[pending].p->op = put_unsafe_op; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pc = pc->nextInst; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->cint.icpc = cglobs->cint.cpc; | 
					
						
							|  |  |  |   cglobs->cint.cpc = opc; | 
					
						
							|  |  |  |   cglobs->cint.BlobsStart = cglobs->cint.CodeStart; | 
					
						
							|  |  |  |   cglobs->cint.CodeStart = OldCodeStart; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | CheckVoids(compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | {				/* establish voids in the head and initial
 | 
					
						
							|  |  |  | 				 * uses        */ | 
					
						
							|  |  |  |   Ventry *ve; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   compiler_vm_op ic; | 
					
						
							|  |  |  |   struct PSEUDO *cpc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cpc = cglobs->cint.CodeStart; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   while ((ic = cpc->op) != allocate_op) { | 
					
						
							|  |  |  |     switch (ic) { | 
					
						
							|  |  |  |     case get_var_op: | 
					
						
							|  |  |  |     case unify_var_op: | 
					
						
							|  |  |  |     case unify_last_var_op: | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  |     case unify_s_var_op: | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     case save_pair_op: | 
					
						
							|  |  |  |     case save_appl_op: | 
					
						
							|  |  |  |       ve = ((Ventry *) cpc->rnd1); | 
					
						
							|  |  |  |       if ((ve->FlagsOfVE & PermFlag) == 0 && ve->RCountOfVE <= 1) { | 
					
						
							|  |  |  | 	ve->NoOfVE = ve->KindOfVE = VoidVar; | 
					
						
							|  |  |  | 	if (ic == get_var_op || ic == | 
					
						
							| 
									
										
										
										
											2001-05-21 20:00:05 +00:00
										 |  |  | 	    save_pair_op || ic == save_appl_op | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  | 	    || ic == unify_s_var_op | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-05-21 20:00:05 +00:00
										 |  |  | 	    ) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  cpc->op = nop_op; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (ic != get_var_op) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     case get_val_op: | 
					
						
							|  |  |  |     case get_atom_op: | 
					
						
							|  |  |  |     case get_num_op: | 
					
						
							|  |  |  |     case get_float_op: | 
					
						
							| 
									
										
										
										
											2010-02-03 18:52:10 +00:00
										 |  |  |     case get_dbterm_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case get_longint_op: | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  |     case get_string_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case get_bigint_op: | 
					
						
							|  |  |  |     case get_list_op: | 
					
						
							|  |  |  |     case get_struct_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs->Uses[cpc->rnd2] = 1; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     cpc = cpc->nextInst; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | checktemp(Int arg, Int rn, compiler_vm_op ic, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   Ventry *v = (Ventry *) arg; | 
					
						
							|  |  |  |   PInstr *q; | 
					
						
							|  |  |  |   Int Needed[MaxTemps]; | 
					
						
							|  |  |  |   Int r, target1, target2; | 
					
						
							|  |  |  |   Int n, *np, *rp; | 
					
						
							|  |  |  |   CELL *cp; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   Int vadr; | 
					
						
							|  |  |  |   Int vreg; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->vadr = vadr = (v->NoOfVE); | 
					
						
							|  |  |  |   cglobs->vreg = vreg = vadr & MaskVarAdrs; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (v->KindOfVE == PermVar || v->KindOfVE == VoidVar) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2001-05-21 20:00:05 +00:00
										 |  |  |   if (v->RCountOfVE == 1) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |      return 0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (vreg) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     --cglobs->Uses[vreg]; | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* follow the life of the variable                                       */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   q = cglobs->cint.cpc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /*
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |    * for(r=0; r<cglobs->MaxCTemps; ++r) Needed[r] = cglobs->Uses[r]; might be written | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |    * as:  | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   np = Needed; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   rp = cglobs->Uses; | 
					
						
							|  |  |  |   for (r = 0; r < cglobs->MaxCTemps; ++r) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     *np++ = *rp++; | 
					
						
							|  |  |  |   if (rn > 0 && (ic == get_var_op || ic == put_var_op)) { | 
					
						
							|  |  |  |     if (ic == put_var_op) | 
					
						
							|  |  |  |       Needed[rn] = 1; | 
					
						
							|  |  |  |     target1 = rn;		/* try to leave it where it is   */ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     target1 = cglobs->MaxCTemps; | 
					
						
							|  |  |  |   target2 = cglobs->MaxCTemps; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   n = v->RCountOfVE - 1; | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  |   while (q != v->LastOpForV && (q = q->nextInst) != NIL) { | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  |     if (q->rnd2 <= 0); /* don't try to reuse REGISTER 0 */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     else if (usesvar(ic = q->op) && arg == q->rnd1) { | 
					
						
							|  |  |  |       --n; | 
					
						
							|  |  |  |       if (ic == put_val_op) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (target1 == cglobs->MaxCTemps && Needed[q->rnd2] == 0) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  target1 = q->rnd2; | 
					
						
							|  |  |  | 	else if (target1 != (r = q->rnd2)) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  if (target2 == cglobs->MaxCTemps && Needed[r] == 0) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    target2 = r; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  else if (target2 > r && cglobs->Uses[r] == 0 && Needed[r] == 0) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    target2 = r; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  |     else if ((ic >= get_var_op && ic <= put_unsafe_op) | 
					
						
							|  |  |  | 	     || ic == get_s_f_op || ic == put_s_f_op) | 
					
						
							|  |  |  |       Needed[q->rnd2] = 1; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     else if (ic >= get_var_op && ic <= put_unsafe_op) | 
					
						
							|  |  |  |       Needed[q->rnd2] = 1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     if ((ic == call_op || ic == safe_call_op) && n == 0) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (target2 < target1) { | 
					
						
							|  |  |  |     r = target2; | 
					
						
							|  |  |  |     target2 = target1; | 
					
						
							|  |  |  |     target1 = r; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   if (target1 == cglobs->MaxCTemps || cglobs->Uses[target1] || Needed[target1]) | 
					
						
							|  |  |  |     if ((target1 = target2) == cglobs->MaxCTemps || cglobs->Uses[target1] || Needed[target1]) { | 
					
						
							|  |  |  |       target1 = cglobs->MaxCTemps; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       do | 
					
						
							|  |  |  | 	--target1; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       while (target1 && cglobs->Uses[target1] == 0 && Needed[target1] == 0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       ++target1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   if (target1 == cglobs->MaxCTemps) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |     CACHE_REGS | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_TYPE = INTERNAL_COMPILER_ERROR; | 
					
						
							|  |  |  |     LOCAL_Error_Term = TermNil; | 
					
						
							|  |  |  |     LOCAL_ErrorMessage = "too many temporaries"; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |     siglongjmp(cglobs->cint.CompilerBotch, COMPILER_ERR_BOTCH); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   v->NoOfVE = cglobs->vadr = vadr = TempVar | target1; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   v->KindOfVE = TempVar; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->Uses[cglobs->vreg = vreg = target1] = v->RCountOfVE - 1; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /*
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |    * for(r=0; r<cglobs->MaxCTemps; ++r) if(cglobs->Contents[r]==vadr) cglobs->Contents[r] = | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |    * NIL;  | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cp = cglobs->Contents; | 
					
						
							|  |  |  |   for (r = 0; r < cglobs->MaxCTemps; ++r) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     if (*cp++ == (Term)vadr) | 
					
						
							|  |  |  |       cp[-1] = NIL; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->Contents[vreg] = vadr; | 
					
						
							|  |  |  |   return 1; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | static Int | 
					
						
							|  |  |  | checkreg(Int arg, Int rn, compiler_vm_op ic, int var_arg, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   PInstr *p = cglobs->cint.cpc; | 
					
						
							|  |  |  |   Int vreg; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (rn >= 0) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     return rn; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (var_arg) { | 
					
						
							|  |  |  |     Ventry *v = (Ventry *) arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vreg = (v->NoOfVE) & MaskVarAdrs; | 
					
						
							|  |  |  |     if (v->KindOfVE == PermVar) | 
					
						
							|  |  |  |       vreg = 0; | 
					
						
							|  |  |  |     else if (vreg == 0) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       checktemp(arg, rn, ic, cglobs); | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |       vreg = (v->NoOfVE) & MaskVarAdrs; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       ++cglobs->Uses[vreg]; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |     if (!vreg) { | 
					
						
							|  |  |  |       vreg = 1; | 
					
						
							|  |  |  |       while (cglobs->Uses[vreg] != 0) { | 
					
						
							|  |  |  | 	++vreg; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       cglobs->Uses[vreg] = v->RCountOfVE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     vreg = 1; | 
					
						
							|  |  |  |     while (cglobs->Uses[vreg] != 0) { | 
					
						
							|  |  |  |       ++vreg; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   while (p) { | 
					
						
							|  |  |  |     if (p->op >= get_var_op && p->op <= put_unsafe_op && p->rnd2 == rn) | 
					
						
							|  |  |  |       p->rnd2 = vreg; | 
					
						
							|  |  |  |     /* only copy variables until you reach a call */ | 
					
						
							|  |  |  |     if (p->op == procceed_op || p->op == call_op || p->op == push_or_op || p->op == pushpop_or_op) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     p = p->nextInst; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   return vreg; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Create a bitmap with all live variables */ | 
					
						
							|  |  |  | static CELL | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | copy_live_temps_bmap(int max, compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-04-08 00:53:38 +01:00
										 |  |  |   unsigned int size = AdjustSize((max|7)/8+1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   int i; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   CELL *dest = Yap_emit_extra_size(mark_live_regs_op, max, size, &cglobs->cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   CELL *ptr=dest; | 
					
						
							|  |  |  |   *ptr = 0L; | 
					
						
							|  |  |  |   for (i=1; i <= max; i++) { | 
					
						
							|  |  |  |     /* move to next cell */ | 
					
						
							|  |  |  |     if (i%(8*CellSize) == 0) { | 
					
						
							|  |  |  |       ptr++; | 
					
						
							|  |  |  |       *ptr = 0L; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* set the register live bit */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     if (cglobs->Contents[i]) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       int j = i%(8*CellSize); | 
					
						
							|  |  |  |       *ptr |= (1<<j); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return((CELL)dest);   | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | c_layout(compiler_struct *cglobs) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-07 15:18:43 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   PInstr *savepc = cglobs->BodyStart->nextInst; | 
					
						
							|  |  |  |   register Ventry *v = cglobs->vtable; | 
					
						
							| 
									
										
										
										
											2011-12-30 16:04:16 +00:00
										 |  |  |   Int *up = cglobs->Uses; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   CELL *cop = cglobs->Contents; | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  |   /* tell put_values used in bip optimisation */ | 
					
						
							|  |  |  |   int rn_kills = 0; | 
					
						
							|  |  |  |   Int rn_to_kill[2]; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |   int needs_either = 0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  |   rn_to_kill[0] = rn_to_kill[1] = 0; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->cint.cpc = cglobs->BodyStart; | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  |   /*
 | 
					
						
							|  |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  |   if (!cglobs->is_a_fact || EAM) { | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   */ | 
					
						
							| 
									
										
										
										
											2005-01-28 23:14:41 +00:00
										 |  |  |   if (!cglobs->is_a_fact) { | 
					
						
							|  |  |  |     while (v != NIL) { | 
					
						
							|  |  |  |       if (v->FlagsOfVE & BranchVar) { | 
					
						
							|  |  |  | 	v->AgeOfVE = v->FirstOfVE + 1;	/* force permanent */ | 
					
						
							|  |  |  | 	++(v->RCountOfVE); | 
					
						
							|  |  |  | 	Yap_emit(put_var_op, (CELL) v, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	v->FlagsOfVE &= ~GlobalVal; | 
					
						
							|  |  |  | 	v->FirstOpForV = cglobs->cint.cpc; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       v = v->NextOfVE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-01-28 23:14:41 +00:00
										 |  |  |     cglobs->cint.cpc->nextInst = savepc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  |     if (cglobs->needs_env || EAM) { | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     if (cglobs->needs_env) { | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  |       LOCAL_nperm = 0; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       AssignPerm(cglobs->cint.CodeStart, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef DEBUG
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       cglobs->pbvars = 0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       CheckUnsafe(cglobs->cint.CodeStart, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef DEBUG
 | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  |       if (cglobs->pbvars != LOCAL_nperm) { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | 	CACHE_REGS | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	LOCAL_Error_TYPE = INTERNAL_COMPILER_ERROR; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = TermNil; | 
					
						
							|  |  |  | 	LOCAL_ErrorMessage = "wrong number of variables found in bitmap"; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  | 	siglongjmp(cglobs->cint.CompilerBotch, OUT_OF_HEAP_BOTCH); | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       }  | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-01-28 23:14:41 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->MaxCTemps = cglobs->nvars + cglobs->max_args - cglobs->tmpreg + cglobs->n_common_exps + 2; | 
					
						
							|  |  |  |   if (cglobs->MaxCTemps >= MaxTemps) | 
					
						
							|  |  |  |     cglobs->MaxCTemps = MaxTemps; | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     Int rn; | 
					
						
							|  |  |  |     for (rn = 0; rn < cglobs->MaxCTemps; ++rn) { | 
					
						
							|  |  |  |       /* cglobs->Uses[rn] = 0; cglobs->Contents[rn] = NIL; */ | 
					
						
							|  |  |  |       *up++ = 0; | 
					
						
							|  |  |  |       *cop++ = NIL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   CheckVoids(cglobs); | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* second scan: allocate registers                                       */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs->cint.cpc = cglobs->cint.CodeStart; | 
					
						
							|  |  |  |   while (cglobs->cint.cpc) { | 
					
						
							|  |  |  |     compiler_vm_op ic = cglobs->cint.cpc->op; | 
					
						
							|  |  |  |     Int arg = cglobs->cint.cpc->rnd1; | 
					
						
							|  |  |  |     Int rn = cglobs->cint.cpc->rnd2; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     switch (ic) { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     case pop_or_op: | 
					
						
							|  |  |  |       if (needs_either) | 
					
						
							|  |  |  | 	needs_either--; | 
					
						
							|  |  |  |     case either_op: | 
					
						
							|  |  |  |       needs_either++; | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2003-11-07 16:31:08 +00:00
										 |  |  | #ifdef TABLING_INNER_CUTS
 | 
					
						
							|  |  |  |     case cut_op: | 
					
						
							|  |  |  |     case cutexit_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs->cut_mark->op = clause_with_cut_op; | 
					
						
							| 
									
										
										
										
											2003-11-07 16:31:08 +00:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2011-04-29 14:59:17 +01:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     case cut_op: | 
					
						
							|  |  |  |     case cutexit_op: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	int i, max; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	max = 0; | 
					
						
							|  |  |  | 	for (i = 1; i < cglobs->MaxCTemps; ++i) { | 
					
						
							|  |  |  | 	  if (cglobs->Contents[i]) max = i; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	cglobs->cint.cpc->ops.opseqt[1] = max; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2003-11-07 16:31:08 +00:00
										 |  |  | #endif /* TABLING_INNER_CUTS */
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case allocate_op: | 
					
						
							|  |  |  |     case deallocate_op: | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       if (!cglobs->needs_env) { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	cglobs->cint.cpc->op = nop_op; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       } else { | 
					
						
							|  |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  | 	PELOCK(51,cglobs->cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	if (is_tabled(cglobs->cint.CurrentPred)) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  cglobs->cint.cpc->op = nop_op; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							| 
									
										
										
										
											2012-02-06 15:11:50 +00:00
										 |  |  | 	  if (cglobs->goalno == 1 && !cglobs->or_found && LOCAL_nperm == 0) | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	    cglobs->cint.cpc->op = nop_op; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  | 	UNLOCK(cglobs->cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case pop_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       ic = (cglobs->cint.cpc->nextInst)->op; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       if (ic >= get_var_op && ic <= put_unsafe_op) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	cglobs->cint.cpc->op = nop_op; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case get_var_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       --cglobs->Uses[rn]; | 
					
						
							|  |  |  |       if (checktemp(arg, rn, ic, cglobs)) { | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  | 	if (cglobs->vreg == rn && !EAM) | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	if (cglobs->vreg == rn) | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  cglobs->cint.cpc->op = nop_op; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |       if (!cglobs->Uses[rn]) | 
					
						
							|  |  |  | 	cglobs->Contents[rn] = cglobs->vadr; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case get_val_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       --cglobs->Uses[rn]; | 
					
						
							|  |  |  |       checktemp(arg, rn, ic, cglobs); | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |       if (!cglobs->Uses[rn]) | 
					
						
							|  |  |  | 	cglobs->Contents[rn] = cglobs->vadr; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case f_0_op: | 
					
						
							|  |  |  |       if (rn_to_kill[0]) --cglobs->Uses[rn_to_kill[0]]; | 
					
						
							|  |  |  |       rn_to_kill[1]=rn_to_kill[0]=0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case f_var_op: | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  |       if (rn_to_kill[0]) --cglobs->Uses[rn_to_kill[0]]; | 
					
						
							|  |  |  |       rn_to_kill[1]=rn_to_kill[0]=0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case unify_var_op: | 
					
						
							|  |  |  |     case unify_val_op: | 
					
						
							|  |  |  |     case unify_last_var_op: | 
					
						
							|  |  |  |     case unify_last_val_op: | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  |     case unify_s_var_op: | 
					
						
							|  |  |  |     case unify_s_val_op: | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     case fetch_args_for_bccall: | 
					
						
							|  |  |  |     case bccall_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       checktemp(arg, rn, ic, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case get_atom_op: | 
					
						
							|  |  |  |     case get_num_op: | 
					
						
							|  |  |  |     case get_float_op: | 
					
						
							|  |  |  |     case get_longint_op: | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  |     case get_string_op: | 
					
						
							| 
									
										
										
										
											2010-02-03 18:52:10 +00:00
										 |  |  |     case get_dbterm_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case get_bigint_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       --cglobs->Uses[rn]; | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |       /* This is not safe if we are in the middle of a disjunction and there
 | 
					
						
							|  |  |  | 	 is something ahead. | 
					
						
							|  |  |  |        */ | 
					
						
							|  |  |  |       if (!cglobs->Uses[rn]) | 
					
						
							|  |  |  | 	cglobs->Contents[rn] = arg; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case get_list_op: | 
					
						
							|  |  |  |     case get_struct_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       --cglobs->Uses[rn]; | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  |       if (!cglobs->Uses[rn]) | 
					
						
							|  |  |  | 	cglobs->Contents[rn] = NIL; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case put_var_op: | 
					
						
							|  |  |  |     case put_unsafe_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       rn = checkreg(arg, rn, ic, TRUE, cglobs); | 
					
						
							|  |  |  |       checktemp(arg, rn, ic, cglobs); | 
					
						
							|  |  |  |       cglobs->Contents[rn] = cglobs->vadr; | 
					
						
							|  |  |  |       ++cglobs->Uses[rn]; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case put_val_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       rn = checkreg(arg, rn, ic, TRUE, cglobs); | 
					
						
							|  |  |  |       checktemp(arg, rn, ic, cglobs); | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  |       if (rn && cglobs->Contents[rn] == (Term)cglobs->vadr && !EAM) | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  |       if (rn && cglobs->Contents[rn] == (Term)cglobs->vadr) | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  cglobs->cint.cpc->op = nop_op; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs->Contents[rn] = cglobs->vadr; | 
					
						
							|  |  |  |       ++cglobs->Uses[rn]; | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  |       if (rn_kills) { | 
					
						
							|  |  |  | 	rn_kills--; | 
					
						
							|  |  |  | 	rn_to_kill[rn_kills]=rn; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case fetch_args_cv_op: | 
					
						
							|  |  |  |     case fetch_args_vc_op: | 
					
						
							| 
									
										
										
										
											2006-11-06 18:35:05 +00:00
										 |  |  |     case fetch_args_iv_op: | 
					
						
							|  |  |  |     case fetch_args_vi_op: | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  |       rn_to_kill[1]=rn_to_kill[0]=0; | 
					
						
							|  |  |  |       if (cglobs->cint.cpc->nextInst && | 
					
						
							|  |  |  | 	  cglobs->cint.cpc->nextInst->op == put_val_op && | 
					
						
							|  |  |  | 	  cglobs->cint.cpc->nextInst->nextInst && | 
					
						
							| 
									
										
										
										
											2009-02-25 00:13:56 +00:00
										 |  |  | 	  (cglobs->cint.cpc->nextInst->nextInst->op == f_var_op || | 
					
						
							|  |  |  | 	   cglobs->cint.cpc->nextInst->nextInst->op == f_0_op) ) | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  | 	rn_kills = 1; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2005-01-14 20:55:16 +00:00
										 |  |  |     case f_val_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  |     case write_s_var_op: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Ventry *ve = (Ventry *) arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((ve->FlagsOfVE & PermFlag) == 0 && ve->RCountOfVE <= 1) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	  cglobs->cint.cpc->op = nop_op; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case write_s_val_op: | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     case write_var_op: | 
					
						
							|  |  |  |     case write_val_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       checktemp(arg, rn, ic, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  |     case put_s_f_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs->Contents[rn] = arg; | 
					
						
							|  |  |  |       ++cglobs->Uses[rn]; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     case put_atom_op: | 
					
						
							|  |  |  |     case put_num_op: | 
					
						
							|  |  |  |     case put_float_op: | 
					
						
							|  |  |  |     case put_longint_op: | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  |     case put_string_op: | 
					
						
							| 
									
										
										
										
											2010-02-03 18:52:10 +00:00
										 |  |  |     case put_dbterm_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case put_bigint_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       rn = checkreg(arg, rn, ic, FALSE, cglobs); | 
					
						
							|  |  |  |       if (cglobs->Contents[rn] == arg) | 
					
						
							|  |  |  | 	cglobs->cint.cpc->op = nop_op; | 
					
						
							|  |  |  |       cglobs->Contents[rn] = arg; | 
					
						
							|  |  |  |       ++cglobs->Uses[rn]; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case put_list_op: | 
					
						
							|  |  |  |     case put_struct_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       rn = checkreg(arg, rn, ic, FALSE, cglobs); | 
					
						
							|  |  |  |       cglobs->Contents[rn] = NIL; | 
					
						
							|  |  |  |       ++cglobs->Uses[rn]; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2003-12-27 00:38:53 +00:00
										 |  |  |     case commit_b_op: | 
					
						
							| 
									
										
										
										
											2003-11-07 16:31:08 +00:00
										 |  |  | #ifdef TABLING_INNER_CUTS
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs->cut_mark->op = clause_with_cut_op; | 
					
						
							| 
									
										
										
										
											2003-11-07 16:31:08 +00:00
										 |  |  | #endif /* TABLING_INNER_CUTS */
 | 
					
						
							|  |  |  |     case save_b_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case patch_b_op:  | 
					
						
							|  |  |  |     case save_appl_op: | 
					
						
							|  |  |  |     case save_pair_op: | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       checktemp(arg, rn, ic, cglobs); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case safe_call_op: | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       /*
 | 
					
						
							|  |  |  | 	vsc: The variables will be in use after this!!!! | 
					
						
							| 
									
										
										
										
											2011-12-30 16:04:16 +00:00
										 |  |  | 	{  | 
					
						
							|  |  |  | 	  UInt Arity = RepPredProp((Prop) arg)->ArityOfPE; | 
					
						
							|  |  |  | 	  for (rn = 1; rn <= Arity; ++rn) | 
					
						
							|  |  |  | 	  --cglobs->Uses[rn]; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case call_op: | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     case orelse_op: | 
					
						
							|  |  |  |     case orlast_op: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	up = cglobs->Uses; | 
					
						
							|  |  |  | 	cop = cglobs->Contents; | 
					
						
							|  |  |  | 	for (rn = 1; rn < cglobs->MaxCTemps; ++rn) { | 
					
						
							|  |  |  | 	  *up++ = *cop++ = NIL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case label_op: | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	up = cglobs->Uses; | 
					
						
							|  |  |  | 	cop = cglobs->Contents; | 
					
						
							| 
									
										
										
										
											2006-07-27 19:04:56 +00:00
										 |  |  | 	for (rn = 0; rn < cglobs->MaxCTemps; ++rn) { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	  if (*cop != (TempVar | rn)) { | 
					
						
							|  |  |  | 	    *up++ = *cop++ = NIL; | 
					
						
							|  |  |  | 	  } else { | 
					
						
							|  |  |  | 	    up++; | 
					
						
							|  |  |  | 	    cop++; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case restore_tmps_and_skip_op: | 
					
						
							|  |  |  |     case restore_tmps_op: | 
					
						
							|  |  |  |       /*
 | 
					
						
							|  |  |  | 	This instruction is required by the garbage collector to find out | 
					
						
							|  |  |  | 	how many temporaries are live right now. It is also useful when | 
					
						
							|  |  |  | 	waking up goals before an either or ! instruction. | 
					
						
							|  |  |  |       */ | 
					
						
							| 
									
										
										
										
											2003-10-17 02:11:21 +00:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	PInstr *mycpc = cglobs->cint.cpc, *oldCodeStart = cglobs->cint.CodeStart; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	int i, max; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* instructions must be placed at BlobsStart */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	cglobs->cint.CodeStart = cglobs->cint.BlobsStart; | 
					
						
							|  |  |  | 	cglobs->cint.cpc = cglobs->cint.icpc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	max = 0; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	for (i = 1; i < cglobs->MaxCTemps; ++i) { | 
					
						
							|  |  |  | 	  if (cglobs->Contents[i]) max = i; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 	Yap_emit(label_op, ++cglobs->labelno, Zero, &cglobs->cint); | 
					
						
							|  |  |  | 	mycpc->rnd1 = cglobs->labelno; | 
					
						
							|  |  |  | 	rn = copy_live_temps_bmap(max, cglobs); | 
					
						
							|  |  |  | 	cglobs->cint.icpc = cglobs->cint.cpc; | 
					
						
							|  |  |  | 	cglobs->cint.BlobsStart = cglobs->cint.CodeStart; | 
					
						
							|  |  |  | 	cglobs->cint.cpc = mycpc; | 
					
						
							|  |  |  | 	cglobs->cint.CodeStart = oldCodeStart; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     default: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     if (cglobs->cint.cpc->nextInst) | 
					
						
							|  |  |  |       cglobs->cint.cpc = cglobs->cint.cpc->nextInst; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     else return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | push_allocate(PInstr *pc, PInstr *oldpc) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /*
 | 
					
						
							|  |  |  |     The idea is to push an allocate forward as much as we can. This | 
					
						
							|  |  |  |     delays work in the emulated code, and in the best case we may get rid of  | 
					
						
							|  |  |  |     allocates altogether. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   /* we can push the allocate */ | 
					
						
							|  |  |  |   int safe = TRUE; | 
					
						
							|  |  |  |   PInstr *initial = oldpc, *dealloc_founds[16]; | 
					
						
							|  |  |  |   int d_founds = 0; | 
					
						
							|  |  |  |   int level = 0; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   while (pc) { | 
					
						
							|  |  |  |     switch (pc->op) { | 
					
						
							|  |  |  |     case jump_op: | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     case call_op: | 
					
						
							|  |  |  |     case safe_call_op: | 
					
						
							|  |  |  |       if (!safe) | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  | 	PInstr *where = initial->nextInst->nextInst; | 
					
						
							|  |  |  | 	while (d_founds) | 
					
						
							|  |  |  | 	  dealloc_founds[--d_founds]->op = nop_op; | 
					
						
							|  |  |  | 	if (where == pc || oldpc == initial->nextInst) | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	oldpc->nextInst = initial->nextInst; | 
					
						
							|  |  |  | 	initial->nextInst->nextInst = pc; | 
					
						
							|  |  |  | 	initial->nextInst = where; | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case push_or_op: | 
					
						
							|  |  |  |       /* we cannot just put an allocate here, because it may never be executed */ | 
					
						
							|  |  |  |       level++; | 
					
						
							|  |  |  |       safe = FALSE; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case pushpop_or_op: | 
					
						
							|  |  |  |       /* last branch and we did not need an allocate so far, cool! */ | 
					
						
							|  |  |  |       level--; | 
					
						
							|  |  |  |       if (!level) | 
					
						
							|  |  |  | 	safe = TRUE; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case cut_op: | 
					
						
							|  |  |  |     case either_op: | 
					
						
							|  |  |  |     case execute_op: | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     case deallocate_op: | 
					
						
							|  |  |  |       dealloc_founds[d_founds++] = pc; | 
					
						
							|  |  |  |       if (d_founds == 16) | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     oldpc = pc; | 
					
						
							|  |  |  |     pc = pc->nextInst; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | c_optimize(PInstr *pc) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char onTail; | 
					
						
							|  |  |  |   Ventry *v; | 
					
						
							|  |  |  |   PInstr *opc = NULL; | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  |   PInstr *inpc = pc; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  |   pc = inpc; | 
					
						
							|  |  |  |   opc = NULL; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* first reverse the pointers */ | 
					
						
							|  |  |  |   while (pc != NULL) { | 
					
						
							|  |  |  |     PInstr *tpc = pc->nextInst; | 
					
						
							|  |  |  |     pc->nextInst = opc; | 
					
						
							|  |  |  |     opc = pc; | 
					
						
							|  |  |  |     pc = tpc; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   pc = opc; | 
					
						
							|  |  |  |   opc = NULL; | 
					
						
							|  |  |  |   onTail = 1; | 
					
						
							|  |  |  |   do { | 
					
						
							|  |  |  |     PInstr *npc = pc->nextInst; | 
					
						
							|  |  |  |     pc->nextInst = opc; | 
					
						
							|  |  |  |     switch (pc->op) { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     case get_var_op: | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  |       /* handle clumsy either branches */ | 
					
						
							|  |  |  |       if (npc->op == f_0_op) { | 
					
						
							|  |  |  | 	npc->rnd1 = pc->rnd1; | 
					
						
							|  |  |  | 	npc->op = f_var_op; | 
					
						
							|  |  |  | 	pc->op = nop_op; | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case put_val_op: | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |     case get_val_op: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Ventry *ve = (Ventry *) pc->rnd1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ve->KindOfVE == TempVar) { | 
					
						
							|  |  |  | 	  UInt argno = ve->NoOfVE & MaskVarAdrs; | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  | 	  if (argno && argno == pc->rnd2) { | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  | 	    pc->op = nop_op; | 
					
						
							|  |  |  | 	  }	   | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       onTail = 1; | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case save_pair_op: | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  Term ve = (Term) pc->rnd1; | 
					
						
							|  |  |  | 	  PInstr *npc = pc->nextInst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (((Ventry *) ve)->RCountOfVE <= 1) | 
					
						
							|  |  |  | 	    pc->op = nop_op; | 
					
						
							|  |  |  | 	  else { | 
					
						
							|  |  |  | 	    *pc = *npc; | 
					
						
							|  |  |  | 	    pc->nextInst = npc; | 
					
						
							|  |  |  | 	    npc->op = save_pair_op; | 
					
						
							|  |  |  | 	    npc->rnd1 = (CELL) ve; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     case save_appl_op: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Term ve = (Term) pc->rnd1; | 
					
						
							|  |  |  | 	PInstr *npc = pc->nextInst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (((Ventry *) ve)->RCountOfVE <= 1) | 
					
						
							|  |  |  | 	  pc->op = nop_op; | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 	  *pc = *npc; | 
					
						
							|  |  |  | 	  pc->nextInst = npc; | 
					
						
							|  |  |  | 	  npc->op = save_appl_op; | 
					
						
							|  |  |  | 	  npc->rnd1 = (CELL) ve; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case nop_op: | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case unify_var_op: | 
					
						
							|  |  |  |     case unify_last_var_op: | 
					
						
							|  |  |  | #ifdef OLD_SYSTEM
 | 
					
						
							|  |  |  |       /* In the good old days Yap would remove lots of small void
 | 
					
						
							|  |  |  |        * instructions for a structure. This is not such a | 
					
						
							|  |  |  |        * good idea nowadays, as we need to know where we | 
					
						
							|  |  |  |        * finish the structure for the last instructions to | 
					
						
							|  |  |  |        * work correctly. Instead, we will use unify_void | 
					
						
							|  |  |  |        * with very little overhead */ | 
					
						
							|  |  |  |       v = (Ventry *) (pc->rnd1); | 
					
						
							|  |  |  |       if (v->KindOfVE == VoidVar && onTail) { | 
					
						
							|  |  |  | 	pc->op = nop_op; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | #endif	/* OLD_SYSTEM */
 | 
					
						
							|  |  |  | 	onTail = 0; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case unify_val_op: | 
					
						
							|  |  |  |       v = (Ventry *) (pc->rnd1); | 
					
						
							|  |  |  |       if (!(v->FlagsOfVE & GlobalVal)) | 
					
						
							|  |  |  | 	pc->op = unify_local_op; | 
					
						
							|  |  |  |       onTail = 0; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case unify_last_val_op: | 
					
						
							|  |  |  |       v = (Ventry *) (pc->rnd1); | 
					
						
							|  |  |  |       if (!(v->FlagsOfVE & GlobalVal)) | 
					
						
							|  |  |  | 	pc->op = unify_last_local_op; | 
					
						
							|  |  |  |       onTail = 0; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case write_val_op: | 
					
						
							|  |  |  |       v = (Ventry *) (pc->rnd1); | 
					
						
							|  |  |  |       if (!(v->FlagsOfVE & GlobalVal)) | 
					
						
							|  |  |  | 	pc->op = write_local_op; | 
					
						
							|  |  |  |       onTail = 0; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case pop_op: | 
					
						
							|  |  |  |       if (FALSE && onTail == 1) { | 
					
						
							|  |  |  | 	pc->op = nop_op; | 
					
						
							|  |  |  | 	onTail = 1; | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  | 	PInstr *p = pc->nextInst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (p != NIL && p->op == nop_op) | 
					
						
							|  |  |  | 	  p = p->nextInst; | 
					
						
							|  |  |  | 	if (p != NIL && p->op == pop_op) { | 
					
						
							|  |  |  | 	  pc->rnd1 += p->rnd1; | 
					
						
							|  |  |  | 	  pc->nextInst = p->nextInst; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	onTail = 2; | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case write_var_op: | 
					
						
							|  |  |  |     case unify_atom_op: | 
					
						
							|  |  |  |     case unify_last_atom_op: | 
					
						
							|  |  |  |     case write_atom_op: | 
					
						
							|  |  |  |     case unify_num_op: | 
					
						
							|  |  |  |     case unify_last_num_op: | 
					
						
							|  |  |  |     case write_num_op: | 
					
						
							|  |  |  |     case unify_float_op: | 
					
						
							|  |  |  |     case unify_last_float_op: | 
					
						
							|  |  |  |     case write_float_op: | 
					
						
							|  |  |  |     case unify_longint_op: | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  |     case unify_string_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case unify_bigint_op: | 
					
						
							|  |  |  |     case unify_last_longint_op: | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  |     case unify_last_string_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case unify_last_bigint_op: | 
					
						
							|  |  |  |     case write_longint_op: | 
					
						
							| 
									
										
										
										
											2013-12-02 14:49:41 +00:00
										 |  |  |     case write_string_op: | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case write_bigint_op: | 
					
						
							|  |  |  |     case unify_list_op: | 
					
						
							|  |  |  |     case write_list_op: | 
					
						
							|  |  |  |     case unify_struct_op: | 
					
						
							|  |  |  |     case write_struct_op: | 
					
						
							|  |  |  |     case write_unsafe_op: | 
					
						
							|  |  |  |     case unify_last_list_op: | 
					
						
							|  |  |  |     case write_last_list_op: | 
					
						
							|  |  |  |     case unify_last_struct_op: | 
					
						
							|  |  |  |     case write_last_struct_op: | 
					
						
							|  |  |  | #ifdef SFUNC
 | 
					
						
							|  |  |  |     case unify_s_f_op: | 
					
						
							|  |  |  |     case write_s_f_op: | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       onTail = 0; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       onTail = 1; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     opc = pc; | 
					
						
							|  |  |  |     pc = npc; | 
					
						
							|  |  |  |   } while (pc != NULL); | 
					
						
							| 
									
										
										
										
											2009-03-05 16:12:21 +00:00
										 |  |  |   pc = inpc; | 
					
						
							|  |  |  |   opc = NULL; | 
					
						
							|  |  |  |   while (pc != NULL) { | 
					
						
							|  |  |  |     if (pc->op == allocate_op) { | 
					
						
							|  |  |  |       push_allocate(pc, opc); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     }  | 
					
						
							|  |  |  |     opc = pc; | 
					
						
							|  |  |  |     pc = pc->nextInst; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-12-27 16:53:09 +00:00
										 |  |  | yamop * | 
					
						
							| 
									
										
										
										
											2010-05-04 14:53:05 +01:00
										 |  |  | Yap_cclause(volatile Term inp_clause, Int NOfArgs, Term mod, volatile Term src) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | {				/* compile a prolog clause, copy of clause myst be in ARG1 */ | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* returns address of code for clause */ | 
					
						
							|  |  |  |   Term head, body; | 
					
						
							| 
									
										
										
										
											2002-12-27 16:53:09 +00:00
										 |  |  |   yamop *acode; | 
					
						
							| 
									
										
										
										
											2004-12-05 05:01:45 +00:00
										 |  |  |   Term my_clause; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   volatile int maxvnum = 512; | 
					
						
							|  |  |  |   int botch_why; | 
					
						
							|  |  |  |   /* may botch while doing a different module */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   /* first, initialise cglobs->cint.CompilerBotch to handle all cases of interruptions */ | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |   compiler_struct cglobs; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |   /* make sure we know there was no error yet */ | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |   LOCAL_ErrorMessage = NULL; | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |   if ((botch_why = sigsetjmp(cglobs.cint.CompilerBotch, 0))) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     restore_machine_regs(); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     reset_vars(cglobs.vtable); | 
					
						
							| 
									
										
										
										
											2010-03-31 15:51:18 +01:00
										 |  |  |     Yap_ReleaseCMem(&cglobs.cint); | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |     switch(botch_why) { | 
					
						
							|  |  |  |     case OUT_OF_STACK_BOTCH: | 
					
						
							|  |  |  |       /* out of local stack, just duplicate the stack */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	Int osize = 2*sizeof(CELL)*(ASP-HR); | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | 	ARG1 = inp_clause; | 
					
						
							|  |  |  | 	ARG3 = src; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	YAPLeaveCriticalSection(); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	if (!Yap_gcl(LOCAL_Error_Size, NOfArgs, ENV, gc_P(P,CP))) { | 
					
						
							|  |  |  | 	  LOCAL_Error_TYPE = OUT_OF_STACK_ERROR; | 
					
						
							|  |  |  | 	  LOCAL_Error_Term = inp_clause; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  | 	if (osize > ASP-HR) { | 
					
						
							|  |  |  | 	  if (!Yap_growstack(2*sizeof(CELL)*(ASP-HR))) { | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  | 	    LOCAL_Error_TYPE = OUT_OF_STACK_ERROR; | 
					
						
							|  |  |  | 	    LOCAL_Error_Term = inp_clause; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	YAPEnterCriticalSection(); | 
					
						
							|  |  |  | 	src = ARG3; | 
					
						
							|  |  |  | 	inp_clause = ARG1; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case OUT_OF_AUX_BOTCH: | 
					
						
							|  |  |  |       /* out of local stack, just duplicate the stack */ | 
					
						
							|  |  |  |       YAPLeaveCriticalSection(); | 
					
						
							|  |  |  |       ARG1 = inp_clause; | 
					
						
							|  |  |  |       ARG3 = src; | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       if (!Yap_ExpandPreAllocCodeSpace(LOCAL_Error_Size, NULL, TRUE)) { | 
					
						
							|  |  |  | 	LOCAL_Error_TYPE = OUT_OF_AUXSPACE_ERROR; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = inp_clause; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       YAPEnterCriticalSection(); | 
					
						
							|  |  |  |       src = ARG3; | 
					
						
							|  |  |  |       inp_clause = ARG1; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case OUT_OF_TEMPS_BOTCH: | 
					
						
							|  |  |  |       /* out of temporary cells */ | 
					
						
							|  |  |  |       if (maxvnum < 16*1024) { | 
					
						
							|  |  |  | 	maxvnum *= 2; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  | 	maxvnum += 4096; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case OUT_OF_HEAP_BOTCH: | 
					
						
							|  |  |  |       /* not enough heap */ | 
					
						
							| 
									
										
										
										
											2004-12-05 05:01:45 +00:00
										 |  |  |       ARG1 = inp_clause; | 
					
						
							|  |  |  |       ARG3 = src; | 
					
						
							| 
									
										
										
										
											2004-06-29 19:04:46 +00:00
										 |  |  |       YAPLeaveCriticalSection(); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       if (!Yap_growheap(FALSE, LOCAL_Error_Size, NULL)) { | 
					
						
							|  |  |  | 	LOCAL_Error_TYPE = OUT_OF_HEAP_ERROR; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = inp_clause; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |       YAPEnterCriticalSection(); | 
					
						
							|  |  |  |       src = ARG3; | 
					
						
							|  |  |  |       inp_clause = ARG1; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case OUT_OF_TRAIL_BOTCH: | 
					
						
							|  |  |  |       /* not enough trail */ | 
					
						
							|  |  |  |       ARG1 = inp_clause; | 
					
						
							|  |  |  |       ARG3 = src; | 
					
						
							|  |  |  |       YAPLeaveCriticalSection(); | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       if (!Yap_growtrail(LOCAL_TrailTop-(ADDR)TR, FALSE)) { | 
					
						
							|  |  |  | 	LOCAL_Error_TYPE = OUT_OF_TRAIL_ERROR; | 
					
						
							|  |  |  | 	LOCAL_Error_Term = inp_clause; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2004-06-29 19:04:46 +00:00
										 |  |  |       YAPEnterCriticalSection(); | 
					
						
							| 
									
										
										
										
											2004-12-05 05:01:45 +00:00
										 |  |  |       src = ARG3; | 
					
						
							|  |  |  |       inp_clause = ARG1; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       return NULL; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2004-12-05 05:01:45 +00:00
										 |  |  |   my_clause = inp_clause; | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   HB = HR; | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |   LOCAL_ErrorMessage = NULL; | 
					
						
							|  |  |  |   LOCAL_Error_Size = 0; | 
					
						
							|  |  |  |   LOCAL_Error_TYPE = YAP_NO_ERROR; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* initialize variables for code generation                              */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |    | 
					
						
							|  |  |  |   cglobs.cint.CodeStart = cglobs.cint.cpc = NULL; | 
					
						
							|  |  |  |   cglobs.cint.BlobsStart = cglobs.cint.icpc = NULL; | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  |   cglobs.cint.dbterml = NULL; | 
					
						
							| 
									
										
										
										
											2010-03-31 15:51:18 +01:00
										 |  |  |   cglobs.cint.blks = NULL; | 
					
						
							| 
									
										
										
										
											2010-04-15 22:49:25 +01:00
										 |  |  |   cglobs.cint.label_offset = NULL; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs.cint.freep = | 
					
						
							|  |  |  |     cglobs.cint.freep0 = | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |     (char *) (HR + maxvnum+(sizeof(Int)/sizeof(CELL))*MaxTemps+MaxTemps); | 
					
						
							| 
									
										
										
										
											2009-07-15 14:27:56 -05:00
										 |  |  |   cglobs.cint.success_handler = 0L; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   if (ASP <= CellPtr (cglobs.cint.freep) + 256) { | 
					
						
							|  |  |  |     cglobs.vtable = NULL; | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_Size = (256+maxvnum)*sizeof(CELL); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     save_machine_regs(); | 
					
						
							| 
									
										
										
										
											2010-12-16 01:22:10 +00:00
										 |  |  |     siglongjmp(cglobs.cint.CompilerBotch,3); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |   cglobs.Uses = (Int *)(HR+maxvnum); | 
					
						
							|  |  |  |   cglobs.Contents = (Term *)(HR+maxvnum+(sizeof(Int)/sizeof(CELL))*MaxTemps); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs.curbranch = cglobs.onbranch = 0; | 
					
						
							|  |  |  |   cglobs.branch_pointer = cglobs.parent_branches; | 
					
						
							|  |  |  |   cglobs.or_found = FALSE; | 
					
						
							|  |  |  |   cglobs.max_args = 0; | 
					
						
							|  |  |  |   cglobs.nvars = 0; | 
					
						
							|  |  |  |   cglobs.tmpreg = 0; | 
					
						
							| 
									
										
										
										
											2005-07-06 15:10:18 +00:00
										 |  |  |   cglobs.needs_env = FALSE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /*
 | 
					
						
							|  |  |  |    * 2000 added to H in case we need to construct call(G) when G is a | 
					
						
							|  |  |  |    * variable used as a goal                                        | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   cglobs.vtable = NULL; | 
					
						
							|  |  |  |   cglobs.common_exps = NULL; | 
					
						
							|  |  |  |   cglobs.n_common_exps = 0; | 
					
						
							|  |  |  |   cglobs.labelno = 0L; | 
					
						
							| 
									
										
										
										
											2005-01-28 23:14:41 +00:00
										 |  |  |   cglobs.is_a_fact = FALSE; | 
					
						
							| 
									
										
										
										
											2005-05-12 03:36:33 +00:00
										 |  |  |   cglobs.hasdbrefs = FALSE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   if (IsVarTerm(my_clause)) { | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_TYPE = INSTANTIATION_ERROR; | 
					
						
							|  |  |  |     LOCAL_Error_Term = my_clause; | 
					
						
							|  |  |  |     LOCAL_ErrorMessage = "in compiling clause"; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (IsApplTerm(my_clause) && FunctorOfTerm(my_clause) == FunctorAssert) { | 
					
						
							|  |  |  |     head = ArgOfTerm(1, my_clause); | 
					
						
							|  |  |  |     body = ArgOfTerm(2, my_clause); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     head = my_clause, body = MkAtomTerm(AtomTrue); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (IsVarTerm(head) || IsPairTerm(head) || IsIntTerm(head) || IsFloatTerm(head) || IsRefTerm(head)) { | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     LOCAL_Error_TYPE = TYPE_ERROR_CALLABLE; | 
					
						
							|  |  |  |     LOCAL_Error_Term = my_clause; | 
					
						
							|  |  |  |     LOCAL_ErrorMessage = "clause should be atom or term"; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     return (0); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     /* find out which predicate we are compiling for */ | 
					
						
							|  |  |  |     if (IsAtomTerm(head)) { | 
					
						
							| 
									
										
										
										
											2001-10-03 13:39:16 +00:00
										 |  |  |       Atom ap = AtomOfTerm(head); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs.cint.CurrentPred = RepPredProp(PredPropByAtom(ap, mod)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |       cglobs.cint.CurrentPred = RepPredProp(PredPropByFunc(FunctorOfTerm(head),mod)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     /* insert extra instructions to count calls */ | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  |     PELOCK(52,cglobs.cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     if ((cglobs.cint.CurrentPred->PredFlags & ProfiledPredFlag) || | 
					
						
							|  |  |  | 	(PROFILING && (cglobs.cint.CurrentPred->cs.p_code.FirstClause == NIL))) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       profiling = TRUE; | 
					
						
							| 
									
										
										
										
											2002-09-03 14:28:09 +00:00
										 |  |  |       call_counting = FALSE; | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     } else if ((cglobs.cint.CurrentPred->PredFlags & CountPredFlag) || | 
					
						
							|  |  |  | 	       (CALL_COUNTING && (cglobs.cint.CurrentPred->cs.p_code.FirstClause == NIL))) { | 
					
						
							| 
									
										
										
										
											2002-09-03 14:28:09 +00:00
										 |  |  |       call_counting = TRUE; | 
					
						
							|  |  |  |       profiling = FALSE; | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       profiling = FALSE; | 
					
						
							| 
									
										
										
										
											2002-09-03 14:28:09 +00:00
										 |  |  |       call_counting = FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  |     UNLOCK(cglobs.cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2005-01-28 23:14:41 +00:00
										 |  |  |   cglobs.is_a_fact = (body == MkAtomTerm(AtomTrue)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* phase 1 : produce skeleton code and variable information              */ | 
					
						
							| 
									
										
										
										
											2006-03-24 17:13:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   c_head(head, &cglobs); | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-01-28 23:14:41 +00:00
										 |  |  |   if (cglobs.is_a_fact && !cglobs.vtable) { | 
					
						
							| 
									
										
										
										
											2005-03-04 20:30:14 +00:00
										 |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2010-07-25 11:22:16 +01:00
										 |  |  |     PELOCK(53,cglobs.cint.CurrentPred); | 
					
						
							| 
									
										
										
										
											2005-03-04 20:30:14 +00:00
										 |  |  |     if (is_tabled(cglobs.cint.CurrentPred)) | 
					
						
							|  |  |  |       Yap_emit(table_new_answer_op, Zero, cglobs.cint.CurrentPred->ArityOfPE, &cglobs.cint); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  | #endif /* TABLING */
 | 
					
						
							|  |  |  |       Yap_emit(procceed_op, Zero, Zero, &cglobs.cint); | 
					
						
							|  |  |  | #ifdef TABLING
 | 
					
						
							| 
									
										
										
										
											2007-11-26 23:43:10 +00:00
										 |  |  |     UNLOCK(cglobs.cint.CurrentPred->PELock); | 
					
						
							| 
									
										
										
										
											2005-03-04 20:30:14 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |     /* ground term, do not need much more work */ | 
					
						
							|  |  |  |     if (cglobs.cint.BlobsStart != NULL) { | 
					
						
							|  |  |  |       cglobs.cint.cpc->nextInst = cglobs.cint.BlobsStart; | 
					
						
							|  |  |  |       cglobs.cint.BlobsStart = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     if (LOCAL_ErrorMessage) | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |       return (0); | 
					
						
							| 
									
										
										
										
											2013-01-15 22:58:34 +00:00
										 |  |  |     /* make sure we give enough space for the  fact */ | 
					
						
							|  |  |  |     if (cglobs.space_op) | 
					
						
							|  |  |  |       cglobs.space_op->rnd1 = cglobs.space_used; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  | #ifdef DEBUG
 | 
					
						
							| 
									
										
										
										
											2011-05-25 16:40:36 +01:00
										 |  |  |     if (GLOBAL_Option['g' - 96]) | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |       Yap_ShowCode(&cglobs.cint); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2003-11-07 16:31:08 +00:00
										 |  |  | #ifdef TABLING_INNER_CUTS
 | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |     Yap_emit(nop_op, Zero, Zero, &cglobs.cint); | 
					
						
							| 
									
										
										
										
											2011-04-29 14:59:17 +01:00
										 |  |  |     cglobs.cut_mark->op = clause_with_cut_op; | 
					
						
							| 
									
										
										
										
											2003-11-07 16:31:08 +00:00
										 |  |  | #endif /* TABLING_INNER_CUTS */
 | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |     Yap_emit(allocate_op, Zero, Zero, &cglobs.cint); | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  |   if (EAM) Yap_emit(body_op, Zero, Zero, &cglobs.cint); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |     c_body(body, mod, &cglobs); | 
					
						
							|  |  |  |     /* Insert blobs at the very end */  | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-02 13:04:30 +01:00
										 |  |  |     if (cglobs.space_op) | 
					
						
							|  |  |  |       cglobs.space_op->rnd1 = cglobs.space_used; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |     if (cglobs.cint.BlobsStart != NULL) { | 
					
						
							|  |  |  |       cglobs.cint.cpc->nextInst = cglobs.cint.BlobsStart; | 
					
						
							|  |  |  |       cglobs.cint.BlobsStart = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |     reset_vars(cglobs.vtable); | 
					
						
							| 
									
										
										
										
											2014-01-19 21:15:05 +00:00
										 |  |  |     HR = HB; | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |     if (B != NULL) { | 
					
						
							|  |  |  |       HB = B->cp_h; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |     if (LOCAL_ErrorMessage) | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |       return (0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef DEBUG
 | 
					
						
							| 
									
										
										
										
											2011-05-25 16:40:36 +01:00
										 |  |  |     if (GLOBAL_Option['g' - 96]) | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |       Yap_ShowCode(&cglobs.cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-03-10 16:27:39 +00:00
										 |  |  |     /* phase 2: classify variables and optimize temporaries          */ | 
					
						
							|  |  |  |     c_layout(&cglobs); | 
					
						
							|  |  |  |     /* Insert blobs at the very end */  | 
					
						
							|  |  |  |     if (cglobs.cint.BlobsStart != NULL) { | 
					
						
							|  |  |  |       cglobs.cint.cpc->nextInst = cglobs.cint.BlobsStart; | 
					
						
							|  |  |  |       cglobs.cint.BlobsStart = NULL; | 
					
						
							|  |  |  |       while (cglobs.cint.cpc->nextInst != NULL) | 
					
						
							|  |  |  | 	cglobs.cint.cpc = cglobs.cint.cpc->nextInst; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* eliminate superfluous pop's and unify_var's                   */ | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |   c_optimize(cglobs.cint.CodeStart); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef DEBUG
 | 
					
						
							| 
									
										
										
										
											2011-05-25 16:40:36 +01:00
										 |  |  |   if (GLOBAL_Option['f' - 96]) | 
					
						
							| 
									
										
										
										
											2004-01-23 02:23:51 +00:00
										 |  |  |     Yap_ShowCode(&cglobs.cint); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef BEAM
 | 
					
						
							| 
									
										
										
										
											2006-04-05 00:16:55 +00:00
										 |  |  |  { | 
					
						
							|  |  |  |    void codigo_eam(compiler_struct *); | 
					
						
							|  |  |  |     | 
					
						
							|  |  |  |    if (EAM) codigo_eam(&cglobs); | 
					
						
							|  |  |  |  } | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* phase 3: assemble code                                                */ | 
					
						
							| 
									
										
										
										
											2010-04-15 11:37:39 +01:00
										 |  |  |  acode = Yap_assemble(ASSEMBLING_CLAUSE, src, cglobs.cint.CurrentPred, (cglobs.is_a_fact && !cglobs.hasdbrefs && !(cglobs.cint.CurrentPred->PredFlags & TabledPredFlag)), &cglobs.cint, cglobs.labelno+1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* check first if there was space for us */ | 
					
						
							| 
									
										
										
										
											2010-03-31 15:51:18 +01:00
										 |  |  |   Yap_ReleaseCMem (&cglobs.cint); | 
					
						
							| 
									
										
										
										
											2003-05-20 19:11:59 +00:00
										 |  |  |   if (acode == NULL) { | 
					
						
							| 
									
										
										
										
											2004-12-20 21:44:58 +00:00
										 |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2003-05-20 19:11:59 +00:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2012-03-09 11:46:34 +00:00
										 |  |  |     return acode; | 
					
						
							| 
									
										
										
										
											2003-05-20 19:11:59 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-09-08 22:06:45 +00:00
										 |  |  | #ifdef BEAM
 | 
					
						
							|  |  |  |   #include "toeam.c"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 |