/************************************************************************* * * * Yap Prolog * * * * Yap Prolog was developed at NCCUP - Universidade do Porto * * * * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997 * * * ************************************************************************** * * * File: index.c * * comments: Indexing a Prolog predicate * * * * Last rev: $Date: 2008-08-07 20:51:21 $,$Author: vsc $ ** * $Log: not supported by cvs2svn $ * Revision 1.202 2008/07/11 17:02:07 vsc * fixes by Bart and Tom: mostly libraries but nasty one in indexing * compilation. * * Revision 1.201 2008/05/10 23:24:11 vsc * fix threads and LU * * Revision 1.200 2008/04/16 17:16:47 vsc * make static_clause only commit to a lause if it is sure that is the true * clause. Otherwise, search for the clause. * * Revision 1.199 2008/04/14 21:20:35 vsc * fixed a bug in static_clause (thanks to Jose Santos) * * Revision 1.198 2008/03/25 16:45:53 vsc * make or-parallelism compile again * * Revision 1.197 2008/02/14 14:35:13 vsc * fixes for indexing code. * * Revision 1.196 2008/01/30 10:35:43 vsc * fix indexing in 64 bits (it would split ints from atoms :( ). * * Revision 1.195 2008/01/24 10:20:42 vsc * clause should not try to discover who is fail. * * Revision 1.194 2008/01/24 00:11:59 vsc * garbage collector was not asking for space. * avoid 0 sized calls to mmap. * * Revision 1.193 2008/01/23 17:57:46 vsc * valgrind it! * enable atom garbage collection. * * Revision 1.192 2007/11/26 23:43:08 vsc * fixes to support threads and assert correctly, even if inefficiently. * * Revision 1.191 2007/11/08 15:52:15 vsc * fix some bugs in new dbterm code. * * Revision 1.190 2007/11/07 09:25:27 vsc * speedup meta-calls * * Revision 1.189 2007/11/06 17:02:12 vsc * compile ground terms away. * * Revision 1.188 2007/10/28 11:23:40 vsc * fix overflow * * Revision 1.187 2007/09/22 08:38:05 vsc * nb_ extra stuff plus an indexing overflow fix. * * Revision 1.186 2007/06/20 13:48:45 vsc * fix bug in index emulator * * Revision 1.185 2007/05/02 11:01:37 vsc * get rid of type punning warnings. * * Revision 1.184 2007/03/26 15:18:43 vsc * debugging and clause/3 over tabled predicates would kill YAP. * * Revision 1.183 2007/03/21 23:23:46 vsc * fix excessive trail cleaning in gc tr overflow. * * Revision 1.182 2007/01/28 14:26:36 vsc * WIN32 support * * Revision 1.181 2007/01/08 08:27:19 vsc * fix restore (Trevor) * make indexing a bit faster on IDB * * Revision 1.180 2006/12/27 01:32:37 vsc * diverse fixes * * Revision 1.179 2006/11/27 17:42:02 vsc * support for UNICODE, and other bug fixes. * * Revision 1.178 2006/11/21 16:21:31 vsc * fix I/O mess * fix spy/reconsult mess * * Revision 1.177 2006/11/15 00:13:36 vsc * fixes for indexing code. * * Revision 1.176 2006/11/08 01:53:08 vsc * avoid generating suspensions on static code. * * Revision 1.175 2006/11/06 18:35:04 vsc * 1estranha * * Revision 1.174 2006/10/25 02:31:07 vsc * fix emulation of trust_logical * * Revision 1.173 2006/10/18 13:47:31 vsc * index.c implementation of trust_logical was decrementing the wrong * cp_tr * * Revision 1.172 2006/10/16 17:12:48 vsc * fixes for threaded version. * * Revision 1.171 2006/10/11 14:53:57 vsc * fix memory leak * fix overflow handling * VS: ---------------------------------------------------------------------- * * Revision 1.170 2006/10/10 14:08:16 vsc * small fixes on threaded implementation. * * Revision 1.169 2006/09/20 20:03:51 vsc * improve indexing on floats * fix sending large lists to DB * * Revision 1.168 2006/05/16 18:37:30 vsc * WIN32 fixes * compiler bug fixes * extend interface * * Revision 1.167 2006/05/02 16:44:11 vsc * avoid uninitialized memory at overflow. * * Revision 1.166 2006/05/02 16:39:06 vsc * bug in indexing code * fix warning messages for write.c * * Revision 1.165 2006/04/27 17:04:08 vsc * don't use <= to compare with block top (libc may not have block header). * * Revision 1.164 2006/04/27 14:10:36 rslopes * *** empty log message *** * * Revision 1.163 2006/04/20 15:28:08 vsc * more graph stuff. * * Revision 1.162 2006/04/12 18:56:50 vsc * fix bug in clause: a trust_me followed by a try should be implemented by * reusing the choice-point. * * Revision 1.161 2006/04/05 00:16:54 vsc * Lots of fixes (check logfile for details * * Revision 1.160 2006/03/24 17:13:41 rslopes * New update to BEAM engine. * BEAM now uses YAP Indexing (JITI) * * Revision 1.159 2006/03/22 20:07:28 vsc * take better care of zombies * * Revision 1.158 2006/03/21 21:30:54 vsc * avoid looking around when expanding for statics too. * * Revision 1.157 2006/03/21 19:20:34 vsc * fix fix on index expansion * * Revision 1.156 2006/03/21 17:11:39 vsc * prevent breakage * * Revision 1.155 2006/03/21 15:06:35 vsc * fixes to handle expansion of dyn amic predicates more efficiently. * * Revision 1.154 2006/03/20 19:51:43 vsc * fix indexing and tabling bugs * * Revision 1.153 2006/02/22 11:55:36 vsc * indexing code would get confused about size of float/1, db_reference1. * * Revision 1.152 2006/02/19 02:55:46 vsc * disable indexing on bigints * * Revision 1.151 2006/01/16 02:57:51 vsc * fix bug with very large integers * fix bug where indexing code was looking at code after a cut. * * Revision 1.150 2005/12/23 00:20:13 vsc * updates to gprof * support for __POWER__ * Try to saveregs before siglongjmp. * * Revision 1.149 2005/12/17 03:25:39 vsc * major changes to support online event-based profiling * improve error discovery and restart on scanner. * * Revision 1.148 2005/11/24 15:33:52 tiagosoares * removed some compilation warnings related to the cut-c code * * Revision 1.147 2005/11/18 18:48:52 tiagosoares * support for executing c code when a cut occurs * * Revision 1.146 2005/10/29 02:21:47 vsc * people should be able to disable indexing. * * Revision 1.145 2005/09/08 22:06:44 rslopes * BEAM for YAP update... * * Revision 1.144 2005/08/17 18:48:35 vsc * fix bug in processing overflows of expand_clauses. * * Revision 1.143 2005/08/02 03:09:50 vsc * fix debugger to do well nonsource predicates. * * Revision 1.142 2005/08/01 15:40:37 ricroc * TABLING NEW: better support for incomplete tabling * * Revision 1.141 2005/07/19 16:54:20 rslopes * fix for older compilers... * * Revision 1.140 2005/07/18 17:41:16 vsc * Yap should respect single argument indexing. * * Revision 1.139 2005/07/06 19:33:53 ricroc * TABLING: answers for completed calls can now be obtained by loading (new *option) or executing (default) them from the trie data structure. * * Revision 1.138 2005/07/05 18:32:32 vsc * ifix some wierd cases in indexing code: * would not look at next argument * problem with pvar as last clause (R Camacho). * * Revision 1.137 2005/06/04 07:27:34 ricroc * long int support for tabling * * Revision 1.136 2005/06/03 08:26:32 ricroc * float support for tabling * * Revision 1.135 2005/06/01 20:25:23 vsc * == and \= should not need a choice-point in -> * * Revision 1.134 2005/06/01 16:42:30 vsc * put switch_list_nl back * * Revision 1.133 2005/06/01 14:02:50 vsc * get_rid of try_me?, retry_me? and trust_me? instructions: they are not * significantly used nowadays. * * Revision 1.132 2005/05/31 20:04:17 vsc * fix cleanup of expand_clauses: make sure we have everything with NULL *afterwards. * * Revision 1.131 2005/05/31 19:42:27 vsc * insert some more slack for indices in LU * Use doubly linked list for LU indices so that updating is less cumbersome. * * Revision 1.130 2005/05/31 04:46:06 vsc * fix expand_index on tabled code. * * Revision 1.129 2005/05/31 02:15:53 vsc * fix SYSTEM_ERROR_INTERNAL messages * * Revision 1.128 2005/05/30 05:26:49 vsc * fix tabling * allow atom gc again for now. * * Revision 1.127 2005/05/27 21:44:00 vsc * Don't try to mess with sequences that don't end with a trust. * A fix for the atom garbage collector actually ignore floats ;-). * * Revision 1.126 2005/05/25 18:58:37 vsc * fix another bug in nth_instance, thanks to Pat Caldon * * Revision 1.125 2005/04/28 14:50:45 vsc * clause should always deref before testing type * * Revision 1.124 2005/04/27 20:09:25 vsc * indexing code could get confused with suspension points * some further improvements on oveflow handling * fix paths in Java makefile * changs to support gibbs sampling in CLP(BN) * * Revision 1.123 2005/04/21 13:53:05 vsc * fix bug with (var(X) -> being interpreted as var(X) by indexing code * * Revision 1.122 2005/04/10 04:01:12 vsc * bug fixes, I hope! * * Revision 1.121 2005/04/07 17:48:54 ricroc * Adding tabling support for mixed strategy evaluation (batched and local *scheduling) * UPDATE: compilation flags -DTABLING_BATCHED_SCHEDULING and *-DTABLING_LOCAL_SCHEDULING removed. To support tabling use -DTABLING in the *Makefile or --enable-tabling in configure. * NEW: yap_flag(tabling_mode,MODE) changes the tabling execution mode of all *tabled predicates to MODE (batched, local or default). * NEW: tabling_mode(PRED,MODE) changes the default tabling execution mode of *predicate PRED to MODE (batched or local). * * Revision 1.120 2005/03/15 18:29:23 vsc * fix GPL * fix idb: stuff in coroutines. * * Revision 1.119 2005/03/04 20:30:12 ricroc * bug fixes for YapTab support * * Revision 1.118 2005/03/01 22:25:08 vsc * fix pruning bug * make DL_MALLOC less enthusiastic about walking through buckets. * * Revision 1.117 2005/02/25 00:09:06 vsc * fix fix, otherwise I'd remove two choice-points :-(. * * Revision 1.116 2005/02/24 21:46:39 vsc * Improve error handling routine, trying to make it more robust. * Improve hole handling in stack expansion * Clause interrpeter was supposed to prune _trust_me * Wrong messages for acos and atanh * * Revision 1.115 2005/02/21 16:50:00 vsc * amd64 fixes * library fixes * * Revision 1.114 2005/01/28 23:14:36 vsc * move to Yap-4.5.7 * Fix clause size * * Revision 1.113 2005/01/15 05:21:36 vsc * fix bug in clause emulator * * Revision 1.112 2004/12/28 22:20:35 vsc * some extra bug fixes for trail overflows: some cannot be recovered that *easily, * some can. * * Revision 1.111 2004/12/21 17:17:15 vsc * miscounting of variable-only clauses in groups might lead to bug in indexing * code. * * Revision 1.110 2004/12/06 04:50:22 vsc * fix bug in removing first clause of a try sequence (lu preds) * * Revision 1.109 2004/12/05 05:01:24 vsc * try to reduce overheads when running with goal expansion enabled. * CLPBN fixes * Handle overflows when allocating big clauses properly. * * Revision 1.108 2004/11/19 22:08:42 vsc * replace SYSTEM_ERROR_INTERNAL by out OUT_OF_WHATEVER_ERROR whenever *appropriate. * * Revision 1.107 2004/11/19 17:14:14 vsc * a few fixes for 64 bit compiling. * * Revision 1.106 2004/11/18 22:32:36 vsc * fix situation where we might assume nonextsing double initialization of C *predicates (use * Hidden Pred Flag). * $host_type was double initialized. * * Revision 1.105 2004/11/04 18:22:32 vsc * don't ever use memory that has been freed (that was done by LU). * generic fixes for WIN32 libraries * * Revision 1.104 2004/10/27 15:56:33 vsc * bug fixes on memory overflows and on clauses :- fail being ignored by clause. * * Revision 1.103 2004/10/22 16:53:19 vsc * bug fixes * * Revision 1.102 2004/10/04 18:56:19 vsc * fixes for thread support * fix indexing bug (serious) * * Revision 1.101 2004/09/30 21:37:41 vsc * fixes for thread support * * Revision 1.100 2004/09/30 19:51:54 vsc * fix overflow from within clause/2 * * Revision 1.99 2004/09/27 20:45:03 vsc * Mega clauses * Fixes to sizeof(expand_clauses) which was being overestimated * Fixes to profiling+indexing * Fixes to reallocation of memory after restoring * Make sure all clauses, even for C, end in _Ystop * Don't reuse space for Streams * Fix Stream_F on StreaNo+1 * * Revision 1.98 2004/09/14 03:30:06 vsc * make sure that condor version always grows trail! * * Revision 1.97 2004/09/03 03:11:09 vsc * memory management fixes * * Revision 1.96 2004/08/27 20:18:52 vsc * more small fixes * * Revision 1.95 2004/08/11 16:14:52 vsc * whole lot of fixes: * - memory leak in indexing * - memory management in WIN32 now supports holes * - extend Yap interface, more support for SWI-Interface * - new predicate mktime in system * - buffer console I/O in WIN32 * * Revision 1.94 2004/07/29 18:15:18 vsc * fix severe bug in indexing of floating point numbers * * Revision 1.93 2004/07/23 19:01:14 vsc * fix bad ref count in expand_clauses when copying indexing block * * Revision 1.92 2004/06/29 19:04:42 vsc * fix multithreaded version * include new version of Ricardo's profiler * new predicat atomic_concat * allow multithreaded-debugging * small fixes * * Revision 1.91 2004/06/17 22:07:23 vsc * bad bug in indexing code. * * Revision 1.90 2004/04/29 03:44:04 vsc * fix bad suspended clause counter * * Revision 1.89 2004/04/27 15:03:43 vsc * more fixes for expand_clauses * * Revision 1.88 2004/04/22 03:24:17 vsc * trust_logical should protect the last clause, otherwise it cannot * jump there. * * Revision 1.87 2004/04/21 04:01:53 vsc * fix bad ordering when inserting second clause * * Revision 1.86 2004/04/20 22:08:23 vsc * fixes for corourining * * Revision 1.85 2004/04/16 19:27:31 vsc * more bug fixes * * Revision 1.84 2004/04/14 19:10:38 vsc * expand_clauses: keep a list of clauses to expand * fix new trail scheme for multi-assignment variables * * Revision 1.83 2004/04/07 22:04:04 vsc * fix memory leaks * * Revision 1.82 2004/03/31 01:02:18 vsc * if number of left-over < 1/5 keep list of clauses to expand around * fix call to stack expander * * Revision 1.81 2004/03/25 02:19:10 pmoura * Removed debugging line to allow compilation. * * Revision 1.80 2004/03/19 11:35:42 vsc * trim_trail for default machine * be more aggressive about try-retry-trust chains. * - handle cases where block starts with a wait * - don't use _killed instructions, just let the thing rot by itself. * * * * *************************************************************************/ #ifdef SCCS static char SccsId[] = "%W% %G%"; #endif /** @file index.c @defgroup Indexing Indexing @ingroup YAPProgramming The indexation mechanism restricts the set of clauses to be tried in a procedure by using information about the status of the instantiated arguments of the goal. These arguments are then used as a key, selecting a restricted set of a clauses from all the clauses forming the procedure. As an example, the two clauses for concatenate: ~~~~~ concatenate([],L,L). concatenate([H|T],A,[H|NT]) :- concatenate(T,A,NT). ~~~~~ If the first argument for the goal is a list, then only the second clause is of interest. If the first argument is the nil atom, the system needs to look only for the first clause. The indexation generates instructions that test the value of the first argument, and then proceed to a selected clause, or group of clauses. Note that if the first argument was a free variable, then both clauses should be tried. In general, indexation will not be useful if the first argument is a free variable. When activating a predicate, a Prolog system needs to store state information. This information, stored in a structure known as choice point or fail point, is necessary when backtracking to other clauses for the predicate. The operations of creating and using a choice point are very expensive, both in the terms of space used and time spent. Creating a choice point is not necessary if there is only a clause for the predicate as there are no clauses to backtrack to. With indexation, this situation is extended: in the example, if the first argument was the atom nil, then only one clause would really be of interest, and it is pointless to create a choice point. This feature is even more useful if the first argument is a list: without indexation, execution would try the first clause, creating a choice point. The clause would fail, the choice point would then be used to restore the previous state of the computation and the second clause would be tried. The code generated by the indexation mechanism would behave much more efficiently: it would test the first argument and see whether it is a list, and then proceed directly to the second clause. An important side effect concerns the use of "cut". In the above example, some programmers would use a "cut" in the first clause just to inform the system that the predicate is not backtrackable and force the removal the choice point just created. As a result, less space is needed but with a great loss in expressive power: the "cut" would prevent some uses of the procedure, like generating lists through backtracking. Of course, with indexation the "cut" becomes useless: the choice point is not even created. Indexation is also very important for predicates with a large number of clauses that are used like tables: ~~~~~ logician(aristoteles,greek). logician(frege,german). logician(russel,english). logician(godel,german). logician(whitehead,english). ~~~~~ An interpreter like C-Prolog, trying to answer the query: ~~~~~ ?- logician(godel,X). ~~~~~ would blindly follow the standard Prolog strategy, trying first the first clause, then the second, the third and finally finding the relevant clause. Also, as there are some more clauses after the important one, a choice point has to be created, even if we know the next clauses will certainly fail. A "cut" would be needed to prevent some possible uses for the procedure, like generating all logicians. In this situation, the indexing mechanism generates instructions that implement a search table. In this table, the value of the first argument would be used as a key for fast search of possibly matching clauses. For the query of the last example, the result of the search would be just the fourth clause, and again there would be no need for a choice point. If the first argument is a complex term, indexation will select clauses just by testing its main functor. However, there is an important exception: if the first argument of a clause is a list, the algorithm also uses the list's head if not a variable. For instance, with the following clauses, ~~~~~ rules([],B,B). rules([n(N)|T],I,O) :- rules_for_noun(N,I,N), rules(T,N,O). rules([v(V)|T],I,O) :- rules_for_verb(V,I,N), rules(T,N,O). rules([q(Q)|T],I,O) :- rules_for_qualifier(Q,I,N), rules(T,N,O). ~~~~~ if the first argument of the goal is a list, its head will be tested, and only the clauses matching it will be tried during execution. Some advice on how to take a good advantage of this mechanism: + Try to make the first argument an input argument. + Try to keep together all clauses whose first argument is not a variable, that will decrease the number of tests since the other clauses are always tried. + Try to avoid predicates having a lot of clauses with the same key. For instance, the procedure: ~~~~~ type(n(mary),person). type(n(john), person). type(n(chair),object). type(v(eat),active). type(v(rest),passive). ~~~~~ becomes more efficient with: ~~~~~ type(n(N),T) :- type_of_noun(N,T). type(v(V),T) :- type_of_verb(V,T). type_of_noun(mary,person). type_of_noun(john,person). type_of_noun(chair,object). type_of_verb(eat,active). type_of_verb(rest,passive). ~~~~~ */ /* * This file compiles and removes the indexation code for the prolog compiler * * Some remarks: *try_me always point to inside the code; * try always points to outside * Algorithm: - fetch info on all clauses - if #clauses =1 return - compute groups: seq of variable only clauses seq: of one or more type instructions bound clauses - sort group - select constant --> type instructions --> count constants --> switch for all arguments: select new argument */ #include #include #include "YapCompile.h" #if DEBUG #include "yapio.h" #endif #include "index.h" #ifndef NULL #define NULL (void *)0 #endif #if HAVE_STRING_H #include #endif #include "cut_c.h" #if defined(YAPOR) || defined(THREADS) #define SET_JLBL(X) jlbl = &(ipc->y_u.X) #else #define SET_JLBL(X) #endif static UInt do_index(ClauseDef *, ClauseDef *, struct intermediates *, UInt, UInt, int, int, CELL *); static UInt do_compound_index(ClauseDef *, ClauseDef *, Term *t, struct intermediates *, UInt, UInt, UInt, UInt, int, int, int, CELL *, int); static UInt do_dbref_index(ClauseDef *, ClauseDef *, Term, struct intermediates *, UInt, UInt, int, int, CELL *); static UInt do_blob_index(ClauseDef *, ClauseDef *, Term, struct intermediates *, UInt, UInt, int, int, CELL *, int); static UInt cleanup_sw_on_clauses(CELL larg, UInt sz, OPCODE ecls) { if (larg & 1) { return sz; } else { yamop *xp = (yamop *)larg; if (xp->opc == ecls) { if (xp->y_u.sssllp.s3 == 1) { UInt nsz = sz + (UInt)(NEXTOP((yamop *)NULL, sssllp)) + xp->y_u.sssllp.s1 * sizeof(yamop *); LOCK(ExpandClausesListLock); if (ExpandClausesFirst == xp) ExpandClausesFirst = xp->y_u.sssllp.snext; if (ExpandClausesLast == xp) { ExpandClausesLast = xp->y_u.sssllp.sprev; } if (xp->y_u.sssllp.sprev) { xp->y_u.sssllp.sprev->y_u.sssllp.snext = xp->y_u.sssllp.snext; } if (xp->y_u.sssllp.snext) { xp->y_u.sssllp.snext->y_u.sssllp.sprev = xp->y_u.sssllp.sprev; } UNLOCK(ExpandClausesListLock); #if DEBUG Yap_ExpandClauses--; Yap_expand_clauses_sz -= (UInt)(NEXTOP((yamop *)NULL, sssllp)) + xp->y_u.sssllp.s1 * sizeof(yamop *); #endif if (xp->y_u.sssllp.p->PredFlags & LogUpdatePredFlag) { Yap_LUIndexSpace_EXT -= (UInt)NEXTOP((yamop *)NULL, sssllp) + xp->y_u.sssllp.s1 * sizeof(yamop *); } else Yap_IndexSpace_EXT -= (UInt)(NEXTOP((yamop *)NULL, sssllp)) + xp->y_u.sssllp.s1 * sizeof(yamop *); Yap_FreeCodeSpace((char *)xp); return nsz; } else { xp->y_u.sssllp.s3--; return sz; } } else { return sz; } } } static UInt recover_from_failed_susp_on_cls(struct intermediates *cint, UInt sz) { /* we have to recover all allocated blocks, just follow the code through. */ struct PSEUDO *cpc = cint->CodeStart; OPCODE ecls = Yap_opcode(_expand_clauses); pred_flags_t log_upd_pred = cint->CurrentPred->PredFlags & LogUpdatePredFlag; while (cpc) { switch (cpc->op) { case enter_lu_op: if (cpc->rnd4) { yamop *code_p = (yamop *)cpc->rnd4; yamop *first = code_p->y_u.Illss.l1; yamop *last = code_p->y_u.Illss.l2; while (first) { yamop *next = first->y_u.OtaLl.n; LogUpdClause *cl = first->y_u.OtaLl.d; cl->ClRefCount--; Yap_FreeCodeSpace((char *)first); if (first == last) break; first = next; } } cpc->rnd4 = Zero; break; case jump_v_op: case jump_nv_op: sz = cleanup_sw_on_clauses(cpc->rnd1, sz, ecls); break; case switch_on_type_op: { TypeSwitch *type_sw = (TypeSwitch *)(cpc->arnds); sz = cleanup_sw_on_clauses(type_sw->PairEntry, sz, ecls); sz = cleanup_sw_on_clauses(type_sw->ConstEntry, sz, ecls); sz = cleanup_sw_on_clauses(type_sw->FuncEntry, sz, ecls); sz = cleanup_sw_on_clauses(type_sw->VarEntry, sz, ecls); } break; case switch_c_op: case if_c_op: { AtomSwiEntry *target = (AtomSwiEntry *)(cpc->rnd2); int cases = cpc->rnd1, i; for (i = 0; i < cases; i++) { sz = cleanup_sw_on_clauses(target[i].u_a.Label, sz, ecls); } if (log_upd_pred) { LogUpdIndex *lcl = ClauseCodeToLogUpdIndex(cpc->rnd2); sz += sizeof(LogUpdIndex) + cases * sizeof(AtomSwiEntry); Yap_LUIndexSpace_SW -= sizeof(LogUpdIndex) + cases * sizeof(AtomSwiEntry); Yap_FreeCodeSpace((char *)lcl); } else { StaticIndex *scl = ClauseCodeToStaticIndex(cpc->rnd2); sz += sizeof(StaticIndex) + cases * sizeof(AtomSwiEntry); Yap_IndexSpace_SW -= sizeof(StaticIndex) + cases * sizeof(AtomSwiEntry); Yap_FreeCodeSpace((char *)scl); } } break; case switch_f_op: case if_f_op: { FuncSwiEntry *target = (FuncSwiEntry *)(cpc->rnd2); int cases = cpc->rnd1, i; for (i = 0; i < cases; i++) { sz = cleanup_sw_on_clauses(target[i].u_f.Label, sz, ecls); } if (log_upd_pred) { LogUpdIndex *lcl = ClauseCodeToLogUpdIndex(cpc->rnd2); sz += sizeof(LogUpdIndex) + cases * sizeof(FuncSwiEntry); Yap_LUIndexSpace_SW -= sizeof(LogUpdIndex) + cases * sizeof(FuncSwiEntry); Yap_FreeCodeSpace((char *)lcl); } else { StaticIndex *scl = ClauseCodeToStaticIndex(cpc->rnd2); Yap_IndexSpace_SW -= sizeof(StaticIndex) + cases * sizeof(FuncSwiEntry); sz += sizeof(StaticIndex) + cases * sizeof(FuncSwiEntry); Yap_FreeCodeSpace((char *)scl); } } break; default: break; } cpc = cpc->nextInst; } Yap_ReleaseCMem(cint); if (cint->code_addr) { Yap_FreeCodeSpace((char *)cint->code_addr); cint->code_addr = NULL; } return sz; } static inline int smaller(Term t1, Term t2) { CELL tg1 = LowTagOf(t1), tg2 = LowTagOf(t2); if (tg1 == tg2) { return t1 < t2; } else return tg1 < tg2; } static inline int smaller_or_eq(Term t1, Term t2) { CELL tg1 = LowTagOf(t1), tg2 = LowTagOf(t2); if (tg1 == tg2) { return t1 <= t2; } else return tg1 < tg2; } static inline void clcpy(ClauseDef *d, ClauseDef *s) { memmove((void *)d, (void *)s, sizeof(ClauseDef)); } static void insort(ClauseDef base[], CELL *p, CELL *q, int my_p) { CELL *j; if (my_p) { p[1] = p[0]; for (j = p; j < q; j += 2) { Term key; Int off = *j; CELL *i; key = base[off].Tag; i = j + 1; /* we are at offset 1 */ while (i > p + 1 && smaller(key, base[i[-2]].Tag)) { i[0] = i[-2]; i -= 2; } i[0] = off; } } else { for (j = p + 2; j < q; j += 2) { Term key; Int off = *j; CELL *i; key = base[off].Tag; i = j; /* we are at offset 1 */ while (i > p && smaller(key, base[i[-2]].Tag)) { i[0] = i[-2]; i -= 2; } i[0] = off; } } } /* copy to a new list of terms */ static void msort(ClauseDef *base, CELL *pt, Int size, int my_p) { if (size > 2) { Int half_size = size / 2; CELL *pt_left, *pt_right, *end_pt, *end_pt_left; int left_p, right_p; if (size < 50) { insort(base, pt, pt + 2 * size, my_p); return; } pt_right = pt + half_size * 2; left_p = my_p ^ 1; right_p = my_p; msort(base, pt, half_size, left_p); msort(base, pt_right, size - half_size, right_p); /* now implement a simple merge routine */ /* pointer to after the end of the list */ end_pt = pt + 2 * size; /* pointer to the element after the last element to the left */ end_pt_left = pt + half_size * 2; /* where is left list */ pt_left = pt + left_p; /* where is right list */ pt_right += right_p; /* where is new list */ pt += my_p; /* while there are elements in the left or right vector do compares */ while (pt_left < end_pt_left && pt_right < end_pt) { /* if the element to the left is larger than the one to the right */ if (smaller_or_eq(base[pt_left[0]].Tag, base[pt_right[0]].Tag)) { /* copy the one to the left */ pt[0] = pt_left[0]; /* and avance the two pointers */ pt += 2; pt_left += 2; } else { /* otherwise, copy the one to the right */ pt[0] = pt_right[0]; pt += 2; pt_right += 2; } } /* if any elements were left in the left vector just copy them */ while (pt_left < end_pt_left) { pt[0] = pt_left[0]; pt += 2; pt_left += 2; } /* if any elements were left in the right vector and they are in the wrong place, just copy them */ if (my_p != right_p) { while (pt_right < end_pt) { pt[0] = pt_right[0]; pt += 2; pt_right += 2; } } } else { if (size > 1 && smaller(base[pt[2]].Tag, base[pt[0]].Tag)) { CELL t = pt[2]; pt[2 + my_p] = pt[0]; pt[my_p] = t; } else if (my_p) { pt[1] = pt[0]; if (size > 1) pt[3] = pt[2]; } } } static void copy_back(ClauseDef *dest, CELL *pt, int max) { /* first need to say that we had no need to make a copy */ int i; CELL *tmp = pt; for (i = 0; i < max; i++) { if (*tmp != i) { ClauseDef cl; int j = i; CELL *pnt = tmp; /* found a chain */ /* make a backup copy */ clcpy(&cl, dest + i); do { /* follow the chain */ int k = *pnt; *pnt = j; /* printf("i=%d, k = %d, j = %d\n",i,j,k); */ if (k == i) { clcpy(dest + j, &cl); break; } else { clcpy(dest + j, dest + k); } pnt = pt + 2 * k; j = k; } while (TRUE); } /* we don't need to do swap */ tmp += 2; } } /* sort a group of clauses by using their tags */ static void sort_group(GroupDef *grp, CELL *top, struct intermediates *cint) { int max = (grp->LastClause - grp->FirstClause) + 1, i; CELL *pt, *base; #if USE_SYSTEM_MALLOC if (!(base = (CELL *)Yap_AllocCodeSpace(2 * max * sizeof(CELL)))) { CACHE_REGS save_machine_regs(); LOCAL_Error_Size = 2 * max * sizeof(CELL); siglongjmp(cint->CompilerBotch, 2); } #else base = top; while (top + 2 * max > (CELL *)LOCAL_TrailTop) { if (!Yap_growtrail(2 * max * CellSize, TRUE)) { LOCAL_Error_Size = 2 * max * CellSize; save_machine_regs(); siglongjmp(cint->CompilerBotch, 4); return; } } #endif pt = base; /* initialize vector */ for (i = 0; i < max; i++) { *pt = i; pt += 2; } #define M_EVEN 0 msort(grp->FirstClause, base, max, M_EVEN); copy_back(grp->FirstClause, base, max); #if USE_SYSTEM_MALLOC Yap_FreeCodeSpace((ADDR)base); #endif } /* add copy to register stack for original reg */ static int init_regcopy(wamreg regs[MAX_REG_COPIES], wamreg copy) { regs[0] = copy; return 1; } /* add copy to register stack for original reg */ static int is_regcopy(wamreg regs[MAX_REG_COPIES], int regs_count, wamreg copy) { int i = 0; while (i < regs_count) { if (regs[i] == copy) { return TRUE; } i++; } /* this copy had overflowed, or it just was not there */ return FALSE; } /* add copy to register stack for original reg */ static int delete_regcopy(wamreg regs[MAX_REG_COPIES], int regs_count, wamreg copy) { int i = 0; while (i < regs_count) { if (regs[i] == copy) { /* we found it */ regs[i] = regs[regs_count - 1]; return regs_count - 1; } i++; } /* this copy had overflowed, or it just was not there */ return regs_count; } /* add copy to register stack for original reg */ static int add_regcopy(wamreg regs[MAX_REG_COPIES], int regs_count, Int source, Int copy) { int i = 0; while (i < regs_count) { if (regs[i] == source) { /* we found it, add new element as last element */ if (regs_count == MAX_REG_COPIES) { return regs_count; } regs[regs_count] = copy; return regs_count + 1; } i++; } /* be careful: we may overwrite an existing copy */ return delete_regcopy(regs, regs_count, copy); } /* add copy to register stack for original reg */ inline static int link_regcopies(wamreg regs[MAX_REG_COPIES], int regs_count, Int c1, Int c2) { int i; for (i = 0; i < regs_count; i++) { if (regs[i] == c1) { return add_regcopy(regs, regs_count, c1, c2); } if (regs[i] == c2) { return add_regcopy(regs, regs_count, c2, c1); } } /* this copy could not be found */ regs_count = delete_regcopy(regs, regs_count, c1); return delete_regcopy(regs, regs_count, c2); } static void add_info(ClauseDef *clause, UInt regno) { wamreg myregs[MAX_REG_COPIES]; int nofregs; yamop *cl; nofregs = init_regcopy(myregs, Yap_regnotoreg(regno)); cl = clause->CurrentCode; #include "findclause.h" } static void add_head_info(ClauseDef *clause, UInt regno) { wamreg iarg = Yap_regnotoreg(regno); yamop *cl = clause->CurrentCode; #include "headclause.h" } static void move_next(ClauseDef *clause, UInt regno) { yamop *cl = clause->CurrentCode; wamreg wreg = Yap_regnotoreg(regno); op_numbers op = Yap_op_from_opcode(cl->opc); switch (op) { #if YAP_JIT case _jit_handler: return; #endif #if THREADS case _unlock_lu: clause->CurrentCode = NEXTOP(cl, e); return; #endif case _p_db_ref_x: case _p_float_x: if (wreg == cl->y_u.xl.x) { clause->CurrentCode = NEXTOP(cl, xl); } return; case _get_list: if (wreg == cl->y_u.x.x) { clause->CurrentCode = NEXTOP(cl, x); } return; case _glist_valx: case _gl_void_vary: case _gl_void_valy: case _gl_void_varx: case _gl_void_valx: case _glist_valy: return; case _get_atom: if (wreg == cl->y_u.xc.x) { clause->CurrentCode = NEXTOP(cl, xc); } return; case _get_2atoms: return; case _get_3atoms: return; case _get_4atoms: return; case _get_5atoms: return; case _get_6atoms: return; /* matching is not guaranteed: case _get_float: case _get_longint: case _get_bigint: */ case _get_struct: if (wreg == cl->y_u.xfa.x) { clause->CurrentCode = NEXTOP(cl, xfa); } default: clause->CurrentCode = clause->Code; return; } } static void add_arg_info(ClauseDef *clause, PredEntry *ap, UInt argno) { yamop *cl; if (ap->ModuleOfPred == IDB_MODULE) { cl = clause->Code; } else { cl = clause->ucd.WorkPC; } while (TRUE) { op_numbers op = Yap_op_from_opcode(cl->opc); switch (op) { case _glist_valx: if (argno == 1) { clause->Tag = (CELL)NULL; return; } argno--; cl = NEXTOP(cl, xx); break; case _gl_void_vary: case _gl_void_valy: case _gl_void_varx: case _gl_void_valx: clause->Tag = (CELL)NULL; return; case _glist_valy: if (argno == 1) { clause->Tag = (CELL)NULL; return; } argno = 2; cl = NEXTOP(cl, yx); break; case _unify_l_x_var: case _unify_l_x_val: case _unify_l_x_loc: case _unify_x_var: case _unify_x_val: case _unify_x_loc: if (argno == 1) { clause->Tag = (CELL)NULL; return; } argno--; case _unify_l_x_var_write: case _unify_l_x_val_write: case _unify_l_x_loc_write: case _unify_x_var_write: case _unify_x_val_write: case _unify_x_loc_write: cl = NEXTOP(cl, ox); break; case _save_pair_x_write: case _save_pair_x: case _save_appl_x_write: case _save_appl_x: cl = NEXTOP(cl, ox); break; case _unify_l_x_var2: case _unify_x_var2: if (argno == 1 || argno == 2) { clause->Tag = (CELL)NULL; return; } argno -= 2; case _unify_l_x_var2_write: case _unify_x_var2_write: cl = NEXTOP(cl, oxx); break; case _unify_y_var: case _unify_y_val: case _unify_y_loc: case _unify_l_y_var: case _unify_l_y_val: case _unify_l_y_loc: /* we're just done with the head of a list, but there is nothing inside. */ if (argno == 1) { clause->Tag = (CELL)NULL; return; } argno--; case _unify_y_var_write: case _unify_y_val_write: case _unify_y_loc_write: case _unify_l_y_var_write: case _unify_l_y_val_write: case _unify_l_y_loc_write: cl = NEXTOP(cl, oy); break; case _save_pair_y_write: case _save_pair_y: case _save_appl_y_write: case _save_appl_y: cl = NEXTOP(cl, oy); break; case _unify_l_void: case _unify_void: if (argno == 1) { clause->Tag = (CELL)NULL; return; } argno--; case _unify_l_void_write: case _unify_void_write: cl = NEXTOP(cl, o); break; case _unify_list: case _unify_l_list: if (argno == 1) { clause->Tag = AbsPair(NULL); clause->ucd.WorkPC = NEXTOP(cl, o); return; } argno += 1; /* 2-1: have two extra arguments to skip */ case _unify_list_write: case _unify_l_list_write: cl = NEXTOP(cl, o); break; case _unify_n_voids: case _unify_l_n_voids: if (argno <= cl->y_u.os.s) { clause->Tag = (CELL)NULL; return; } argno -= cl->y_u.os.s; case _unify_n_voids_write: case _unify_l_n_voids_write: cl = NEXTOP(cl, os); break; case _unify_atom: case _unify_l_atom: if (argno == 1) { clause->Tag = cl->y_u.oc.c; return; } argno--; case _unify_atom_write: case _unify_l_atom_write: cl = NEXTOP(cl, oc); break; case _unify_float_write: case _unify_l_float_write: cl = NEXTOP(cl, od); break; case _unify_float: case _unify_l_float: if (argno == 1) { clause->Tag = AbsAppl((CELL *)FunctorDouble); clause->ucd.t_ptr = AbsAppl(cl->y_u.od.d); return; } cl = NEXTOP(cl, od); argno--; break; case _unify_longint: case _unify_l_longint: if (argno == 1) { clause->Tag = AbsAppl((CELL *)FunctorLongInt); clause->ucd.t_ptr = AbsAppl(cl->y_u.oi.i); return; } argno--; cl = NEXTOP(cl, oi); break; case _unify_bigint: case _unify_l_bigint: if (argno == 1) { clause->Tag = AbsAppl((CELL *)FunctorBigInt); clause->ucd.t_ptr = cl->y_u.oc.c; return; } cl = NEXTOP(cl, oc); argno--; break; case _unify_string: case _unify_l_string: if (argno == 1) { clause->Tag = AbsAppl((CELL *)FunctorString); clause->ucd.t_ptr = cl->y_u.ou.ut; return; } cl = NEXTOP(cl, ou); argno--; break; case _unify_n_atoms: if (argno <= cl->y_u.osc.s) { clause->Tag = cl->y_u.osc.c; return; } argno -= cl->y_u.osc.s; case _unify_n_atoms_write: cl = NEXTOP(cl, osc); break; case _unify_struct: case _unify_l_struc: if (argno == 1) { clause->Tag = AbsAppl((CELL *)cl->y_u.ofa.f); clause->ucd.WorkPC = NEXTOP(cl, ofa); return; } /* must skip next n arguments */ argno += cl->y_u.ofa.a - 1; case _unify_l_struc_write: case _unify_struct_write: cl = NEXTOP(cl, ofa); break; case _pop: cl = NEXTOP(cl, e); break; case _pop_n: cl = NEXTOP(cl, s); break; #ifdef BEAM case _run_eam: cl = NEXTOP(cl, os); break; #endif #ifdef THREADS case _unlock_lu: cl = NEXTOP(cl, e); break; #endif case _get_dbterm: cl = NEXTOP(cl, xc); break; case _unify_dbterm: case _unify_l_dbterm: cl = NEXTOP(cl, oc); break; case _unify_idb_term: case _copy_idb_term: { Term t = clause->ucd.c_sreg[argno]; if (IsVarTerm(t)) { clause->Tag = (CELL)NULL; } else if (IsApplTerm(t)) { CELL *pt = RepAppl(t); clause->Tag = AbsAppl((CELL *)pt[0]); if (IsExtensionFunctor(FunctorOfTerm(t))) { clause->ucd.t_ptr = t; } else { clause->ucd.c_sreg = pt; } } else if (IsPairTerm(t)) { CELL *pt = RepPair(t); clause->Tag = AbsPair(NULL); clause->ucd.c_sreg = pt - 1; } else { clause->Tag = t; } } return; default: return; } } } static void skip_to_arg(ClauseDef *clause, PredEntry *ap, UInt argno, int at_point) { yamop *cl; int done = FALSE; if (ap->ModuleOfPred == IDB_MODULE) { return; } else { cl = clause->CurrentCode; } if (!at_point) { clause->CurrentCode = clause->Code; return; } while (!done) { op_numbers op = Yap_op_from_opcode(cl->opc); switch (op) { #ifdef BEAM case _run_eam: clause->CurrentCode = clause->Code; return; #endif case _unify_void: if (argno == 1) { clause->CurrentCode = clause->Code; return; } else { argno--; } case _unify_void_write: cl = NEXTOP(cl, o); break; case _unify_list: case _unify_l_list: case _unify_atom: case _unify_l_atom: /* unification is not guaranteed case _unify_longint: case _unify_l_longint: case _unify_bigint: case _unify_l_bigint: case _unify_l_float: */ case _unify_struct: case _unify_l_struc: if (cl == clause->ucd.WorkPC) { clause->CurrentCode = cl; } else { clause->CurrentCode = clause->Code; } return; case _unify_list_write: case _unify_l_list_write: cl = NEXTOP(cl, o); break; case _unify_n_voids: case _unify_l_n_voids: if (argno <= cl->y_u.os.s) { clause->CurrentCode = clause->Code; return; } else { argno -= cl->y_u.os.s; } case _unify_n_voids_write: case _unify_l_n_voids_write: cl = NEXTOP(cl, os); break; case _unify_atom_write: case _unify_l_atom_write: cl = NEXTOP(cl, oc); break; case _unify_float_write: case _unify_l_float_write: cl = NEXTOP(cl, od); break; case _unify_l_struc_write: case _unify_struct_write: cl = NEXTOP(cl, ofa); break; #ifdef THREADS case _unlock_lu: #endif case _pop: cl = NEXTOP(cl, e); break; case _pop_n: cl = NEXTOP(cl, s); break; default: clause->CurrentCode = clause->Code; return; } } } static UInt groups_in(ClauseDef *min, ClauseDef *max, GroupDef *grp, struct intermediates *cint) { CACHE_REGS UInt groups = 0; while (min <= max) { grp->FirstClause = min; grp->AtomClauses = 0; grp->PairClauses = 0; grp->StructClauses = 0; grp->TestClauses = 0; if (min->Tag == (_var + 1) * sizeof(CELL)) { min++; continue; } /* only do this for the first clauses in a group */ if (IsVarTerm(min->Tag)) { ClauseDef *clp = min + 1; grp->VarClauses = 1; do { if (clp > max || !IsVarTerm(clp->Tag)) { grp->LastClause = (min = clp) - 1; break; } if (clp->Tag != (_var + 1) * sizeof(CELL)) grp->VarClauses++; clp++; } while (TRUE); } else { grp->VarClauses = 0; do { restart_loop: if (IsAtomTerm(min->Tag) || IsIntTerm(min->Tag)) { grp->AtomClauses++; } else if (IsPairTerm(min->Tag)) { grp->PairClauses++; } else if (IsApplTerm(min->Tag)) { grp->StructClauses++; } else { grp->TestClauses++; } min++; } while (min <= max && (!IsVarTerm(min->Tag))); if (min <= max && min->Tag == (_var + 1) * sizeof(CELL)) { min++; if (min < max) goto restart_loop; } grp->LastClause = min - 1; } groups++; grp++; while (grp + 16 > (GroupDef *)LOCAL_TrailTop) { UInt sz = (groups + 16) * sizeof(GroupDef); #if USE_SYSTEM_MALLOC LOCAL_Error_Size = sz; /* grow stack */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 4); #else if (!Yap_growtrail(sz, TRUE)) { LOCAL_Error_Size = sz; save_machine_regs(); siglongjmp(cint->CompilerBotch, 4); return 0; } #endif } } return groups; } static UInt new_label(struct intermediates *cint) { UInt lbl = cint->i_labelno; cint->i_labelno += 2; return lbl; } static Int has_cut(yamop *pc, PredEntry *ap) { if (ap->PredFlags & LogUpdatePredFlag) { LogUpdClause *lcl = ClauseCodeToLogUpdClause(pc); return ((lcl->ClFlags & HasCutMask) != 0); } else if (ap->PredFlags & MegaClausePredFlag) { /* must be a fact */ return FALSE; } else { StaticClause *scl; scl = ClauseCodeToStaticClause(pc); return ((scl->ClFlags & HasCutMask) != 0); } } static void emit_trust(ClauseDef *cl, struct intermediates *cint, UInt nxtlbl, int clauses) { PredEntry *ap = cint->CurrentPred; yamop *clcode = cl->Code; if (ap->PredFlags & TabledPredFlag) clcode = NEXTOP(clcode, Otapl); if (!(ap->PredFlags & LogUpdatePredFlag)) { /* this should not be generated for logical update predicates!! */ if (ap->PredFlags & ProfiledPredFlag) { Yap_emit(retry_profiled_op, Unsigned(ap), Zero, cint); } if (ap->PredFlags & CountPredFlag) { Yap_emit(count_retry_op, Unsigned(ap), Zero, cint); } } if (clauses == 0) { Yap_emit(trust_op, (CELL)clcode, has_cut(cl->Code, ap), cint); } else { Yap_emit(retry_op, (CELL)clcode, (clauses << 1) | has_cut(cl->Code, ap), cint); Yap_emit(jumpi_op, nxtlbl, Zero, cint); } } static void emit_retry(ClauseDef *cl, struct intermediates *cint, int clauses) { PredEntry *ap = cint->CurrentPred; yamop *clcode = cl->Code; if (ap->PredFlags & TabledPredFlag) clcode = NEXTOP(clcode, Otapl); if (!(ap->PredFlags & LogUpdatePredFlag)) { /* this should not be generated for logical update predicates!! */ if (ap->PredFlags & ProfiledPredFlag) { Yap_emit(retry_profiled_op, Unsigned(ap), Zero, cint); } if (ap->PredFlags & CountPredFlag) { Yap_emit(count_retry_op, Unsigned(ap), Zero, cint); } } Yap_emit(retry_op, (CELL)clcode, (clauses << 1) | has_cut(cl->Code, ap), cint); } static compiler_vm_op emit_optry(int var_group, int first, int clauses, int clleft, PredEntry *ap) { /* var group */ if (var_group || clauses == 0) { if (first) { return try_op; } else if (clleft + clauses) { return retry_op; } else { return trust_op; } } else if (clleft == 0) { #ifdef TABLING if (ap->PredFlags & TabledPredFlag && !first) { /* we never actually get to remove the last choice-point in this case */ return retry_op; } else #endif /* TABLING */ { /* last group */ return try_op; } } else { /* nonvar group */ return try_in_op; } } static void emit_try(ClauseDef *cl, struct intermediates *cint, int var_group, int first, int clauses, int clleft, UInt nxtlbl) { PredEntry *ap = cint->CurrentPred; yamop *clcode; compiler_vm_op comp_op; if (ap->PredFlags & LogUpdatePredFlag) { clcode = cl->Code; } else if (ap->PredFlags & TabledPredFlag) { clcode = NEXTOP(cl->Code, Otapl); } else { clcode = cl->CurrentCode; } comp_op = emit_optry(var_group, first, clauses, clleft, cint->CurrentPred); Yap_emit(comp_op, (CELL)clcode, ((clauses + clleft) << 1) | has_cut(cl->Code, ap), cint); } static TypeSwitch *emit_type_switch(compiler_vm_op op, struct intermediates *cint) { return (TypeSwitch *)Yap_emit_extra_size(op, 0, sizeof(TypeSwitch), cint); } static yamop *emit_switch_space(UInt n, UInt item_size, struct intermediates *cint, CELL func_mask) { CACHE_REGS PredEntry *ap = cint->CurrentPred; if (ap->PredFlags & LogUpdatePredFlag) { UInt sz = sizeof(LogUpdIndex) + n * item_size; LogUpdIndex *cl = (LogUpdIndex *)Yap_AllocCodeSpace(sz); if (cl == NULL) { LOCAL_Error_Size = sz; /* grow stack */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 2); } Yap_LUIndexSpace_SW += sz; cl->ClFlags = SwitchTableMask | LogUpdMask | func_mask; cl->ClSize = sz; cl->ClPred = cint->CurrentPred; /* insert into code chain */ Yap_inform_profiler_of_clause(cl, (CODEADDR)cl + sz, ap, GPROF_NEW_LU_SWITCH); return cl->ClCode; } else { UInt sz = sizeof(StaticIndex) + n * item_size; StaticIndex *cl = (StaticIndex *)Yap_AllocCodeSpace(sz); if (cl == NULL) { LOCAL_Error_Size = sz; /* grow stack */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 2); } Yap_IndexSpace_SW += sz; cl->ClFlags = SwitchTableMask; cl->ClSize = sz; cl->ClPred = cint->CurrentPred; Yap_inform_profiler_of_clause(cl, (CODEADDR)cl + sz, ap, GPROF_NEW_STATIC_SWITCH); return cl->ClCode; /* insert into code chain */ } } static AtomSwiEntry *emit_cswitch(COUNT n, yamop *fail_l, struct intermediates *cint) { compiler_vm_op op; AtomSwiEntry *target; if (n > MIN_HASH_ENTRIES) { COUNT cases = MIN_HASH_ENTRIES, i; n += 1 + n / 4; while (cases < n) cases *= 2; n = cases; op = switch_c_op; target = (AtomSwiEntry *)emit_switch_space(n, sizeof(AtomSwiEntry), cint, 0); for (i = 0; i < n; i++) { target[i].Tag = Zero; target[i].u_a.labp = fail_l; } Yap_emit(op, Unsigned(n), (CELL)target, cint); } else { UInt i; op = if_c_op; target = (AtomSwiEntry *)emit_switch_space(n + 1, sizeof(AtomSwiEntry), cint, 0); for (i = 0; i < n; i++) { target[i].u_a.labp = fail_l; } target[n].Tag = Zero; target[n].u_a.labp = fail_l; Yap_emit(op, Unsigned(n), (CELL)target, cint); } return target; } static AtomSwiEntry *lookup_c_hash(Term t, yamop *tab, COUNT entries) { AtomSwiEntry *cebase = (AtomSwiEntry *)tab; int hash, d; AtomSwiEntry *centry; hash = (t >> HASH_SHIFT) & (entries - 1); centry = cebase + hash; d = (entries - 1) & (t | 1); while (centry->Tag != t) { if (centry->Tag == 0L) return centry; hash = (hash + d) & (entries - 1); centry = cebase + hash; } return centry; } static AtomSwiEntry *fetch_centry(AtomSwiEntry *cebase, Term wt, int i, int n) { if (n > MIN_HASH_ENTRIES) { int cases = MIN_HASH_ENTRIES; n += 1 + n / 4; while (cases < n) cases *= 2; return lookup_c_hash(wt, (yamop *)cebase, cases); } else { return cebase + i; } } static FuncSwiEntry *emit_fswitch(COUNT n, yamop *fail_l, struct intermediates *cint) { compiler_vm_op op; FuncSwiEntry *target; if (n > MIN_HASH_ENTRIES) { int cases = MIN_HASH_ENTRIES, i; n += 1 + n / 4; while (cases < n) cases *= 2; n = cases; op = switch_f_op; target = (FuncSwiEntry *)emit_switch_space(n, sizeof(FuncSwiEntry), cint, FuncSwitchMask); for (i = 0; i < n; i++) { target[i].Tag = NULL; target[i].u_f.labp = fail_l; } Yap_emit(op, Unsigned(n), (CELL)target, cint); } else { UInt i; op = if_f_op; target = (FuncSwiEntry *)emit_switch_space(n + 1, sizeof(FuncSwiEntry), cint, FuncSwitchMask); for (i = 0; i < n; i++) { target[i].u_f.labp = fail_l; } target[n].Tag = NULL; target[n].u_f.labp = fail_l; Yap_emit(op, Unsigned(n), (CELL)target, cint); } return target; } static FuncSwiEntry *lookup_f_hash(Functor f, yamop *tab, COUNT entries) { FuncSwiEntry *febase = (FuncSwiEntry *)tab; int hash, d; FuncSwiEntry *fentry; Term wt = (Term)f; hash = (wt >> HASH_SHIFT) & (entries - 1); fentry = febase + hash; d = (entries - 1) & (wt | 1); while (fentry->Tag != f) { if (fentry->Tag == NULL) return fentry; hash = (hash + d) & (entries - 1); fentry = febase + hash; } return fentry; } static FuncSwiEntry *fetch_fentry(FuncSwiEntry *febase, Functor ft, int i, int n) { if (n > MIN_HASH_ENTRIES) { int cases = MIN_HASH_ENTRIES; n += 1 + n / 4; while (cases < n) cases *= 2; return lookup_f_hash(ft, (yamop *)febase, cases); } else { return febase + i; } } /* we assume there is at least one clause, that is, c0 < cf */ static UInt do_var_clauses(ClauseDef *c0, ClauseDef *cf, int var_group, struct intermediates *cint, int first, int clleft, UInt nxtlbl, UInt argno0) { UInt labl; UInt labl_dyn0 = 0, labl_dynf = 0; labl = new_label(cint); Yap_emit(label_op, labl, Zero, cint); /* add expand_node if var_group == TRUE (jump on var) || var_group == FALSE (leaf node) */ if (first && cint->CurrentPred->PredFlags & LogUpdatePredFlag) { UInt ncls; labl_dyn0 = new_label(cint); if (clleft) labl_dynf = labl_dyn0; else labl_dynf = new_label(cint); if (clleft == 0) /* trust*/ ncls = (cf - c0) + 1; else ncls = 0; Yap_emit_4ops(enter_lu_op, labl_dyn0, labl_dynf, ncls, Zero, cint); Yap_emit(label_op, labl_dyn0, Zero, cint); } if (c0 == cf) { emit_try(c0, cint, var_group, first, 0, clleft, nxtlbl); } else { if (c0 < cf) { emit_try(c0, cint, var_group, first, cf - c0, clleft, nxtlbl); } c0++; while (c0 < cf) { emit_retry(c0, cint, clleft + (cf - c0)); c0++; } if (c0 == cf) { emit_trust(c0, cint, nxtlbl, clleft); if (!clleft && cint->CurrentPred->PredFlags & LogUpdatePredFlag) { Yap_emit(label_op, labl_dynf, Zero, cint); } } } return labl; } static UInt do_var_group(GroupDef *grp, struct intermediates *cint, int var_group, int first, int clleft, UInt nxtlbl, UInt argno0) { return do_var_clauses(grp->FirstClause, grp->LastClause, var_group, cint, first, clleft, nxtlbl, argno0); } /* count the number of different constants */ static UInt count_consts(GroupDef *grp) { Term current = MkAtomTerm(AtomFoundVar); UInt i = 0; ClauseDef *cl = grp->FirstClause; while (IsAtomTerm(cl->Tag) || IsIntTerm(cl->Tag)) { if (current != cl->Tag) { i++; current = cl->Tag; } if (cl == grp->LastClause) { return i; } cl++; } return i; } static UInt count_blobs(GroupDef *grp) { UInt i = 1; ClauseDef *cl = grp->FirstClause + 1; Term current = grp->FirstClause->Tag; while (cl <= grp->LastClause) { if (current != cl->Tag) { i++; current = cl->Tag; } cl++; } return i; } /* count the number of different constants */ static UInt count_funcs(GroupDef *grp) { Term current = MkAtomTerm(AtomFoundVar); UInt i = 0; ClauseDef *cl = grp->FirstClause; while (IsApplTerm(cl->Tag)) { if (current != cl->Tag) { i++; current = cl->Tag; } if (cl == grp->LastClause) { return i; } cl++; } return i; } static UInt emit_single_switch_case(ClauseDef *min, struct intermediates *cint, int first, int clleft, UInt nxtlbl) { if (cint->CurrentPred->PredFlags & TabledPredFlag) { /* with tabling we don't clean trust at the very end of computation. */ if (clleft || !first) { /* if we still have clauses left, means we already created a CP, so I should avoid creating again */ return (UInt)NEXTOP(min->Code, Otapl); } else { return (UInt)min->Code; } } if (cint->CurrentPred->PredFlags & LogUpdatePredFlag) { return (UInt)(min->Code); } else { return (UInt)(min->CurrentCode); } } static UInt suspend_indexing(ClauseDef *min, ClauseDef *max, PredEntry *ap, struct intermediates *cint) { UInt tcls = ap->cs.p_code.NOfClauses; UInt cls = (max - min) + 1; if (cint->expand_block && cint->expand_block != (yamop *)(&(ap->cs.p_code.ExpandCode)) && cint->expand_block->y_u.sssllp.s2 < 2 * (max - min)) { cint->expand_block->y_u.sssllp.s3++; return (UInt)(cint->expand_block); } if (cls < tcls / 8) { yamop *ncode; yamop **st; UInt tels; UInt sz; if (ap->PredFlags & LogUpdatePredFlag) { /* give it some slack */ tels = cls + 4; } else { tels = cls+1; } sz = (UInt)NEXTOP((yamop *)NULL, sssllp) + tels * sizeof(yamop *); if ((ncode = (yamop *)Yap_AllocCodeSpace(sz)) == NULL) { save_machine_regs(); siglongjmp(cint->CompilerBotch, 2); } #if DEBUG Yap_ExpandClauses++; Yap_expand_clauses_sz += sz; #endif if (ap->PredFlags & LogUpdatePredFlag) { Yap_LUIndexSpace_EXT += sz; } else { Yap_IndexSpace_EXT += sz; } Yap_inform_profiler_of_clause(ncode, (CODEADDR)ncode + sz, ap, GPROF_NEW_EXPAND_BLOCK); /* create an expand_block */ ncode->opc = Yap_opcode(_expand_clauses); ncode->y_u.sssllp.p = ap; ncode->y_u.sssllp.s1 = tels; ncode->y_u.sssllp.s2 = cls; ncode->y_u.sssllp.s3 = 1; st = (yamop **)NEXTOP(ncode, sssllp); while (min <= max) { *st++ = min->Code; min++; } while (cls < tels) { *st++ = NULL; cls++; } LOCK(ExpandClausesListLock); ncode->y_u.sssllp.snext = ExpandClausesFirst; ncode->y_u.sssllp.sprev = NULL; if (ExpandClausesFirst) ExpandClausesFirst->y_u.sssllp.sprev = ncode; ExpandClausesFirst = ncode; if (ExpandClausesLast == NULL) ExpandClausesLast = ncode; UNLOCK(ExpandClausesListLock); return (UInt)ncode; } return (UInt) & (ap->cs.p_code.ExpandCode); } static void recover_ecls_block(yamop *ipc) { ipc->y_u.sssllp.s3--; if (!ipc->y_u.sssllp.s3) { LOCK(ExpandClausesListLock); if (ExpandClausesFirst == ipc) ExpandClausesFirst = ipc->y_u.sssllp.snext; if (ExpandClausesLast == ipc) { ExpandClausesLast = ipc->y_u.sssllp.sprev; } if (ipc->y_u.sssllp.sprev) { ipc->y_u.sssllp.sprev->y_u.sssllp.snext = ipc->y_u.sssllp.snext; } if (ipc->y_u.sssllp.snext) { ipc->y_u.sssllp.snext->y_u.sssllp.sprev = ipc->y_u.sssllp.sprev; } UNLOCK(ExpandClausesListLock); #if DEBUG Yap_ExpandClauses--; Yap_expand_clauses_sz -= (UInt)(NEXTOP((yamop *)NULL, sssllp)) + ipc->y_u.sssllp.s1 * sizeof(yamop *); #endif /* no dangling pointers for gprof */ Yap_InformOfRemoval(ipc); if (ipc->y_u.sssllp.p->PredFlags & LogUpdatePredFlag) { Yap_LUIndexSpace_EXT -= (UInt)NEXTOP((yamop *)NULL, sssllp) + ipc->y_u.sssllp.s1 * sizeof(yamop *); } else Yap_IndexSpace_EXT -= (UInt)NEXTOP((yamop *)NULL, sssllp) + ipc->y_u.sssllp.s1 * sizeof(yamop *); Yap_FreeCodeSpace((char *)ipc); } } static UInt do_var_entries(GroupDef *grp, Term t, struct intermediates *cint, UInt argno, int first, int clleft, UInt nxtlbl) { PredEntry *ap = cint->CurrentPred; if (!IsVarTerm(t) || t != 0L) { return suspend_indexing(grp->FirstClause, grp->LastClause, ap, cint); } return do_var_group(grp, cint, FALSE, first, clleft, nxtlbl, ap->ArityOfPE + 1); } static UInt do_consts(GroupDef *grp, Term t, struct intermediates *cint, int compound_term, CELL *sreg, UInt arity, int last_arg, UInt argno, int first, UInt nxtlbl, int clleft, CELL *top) { COUNT n; ClauseDef *min = grp->FirstClause; COUNT i; UInt lbl; /* generate a switch */ AtomSwiEntry *cs; PredEntry *ap = cint->CurrentPred; if (!IsAtomTerm(min->Tag) && !IsIntTerm(min->Tag)) { /* no clauses, just skip */ return nxtlbl; } n = count_consts(grp); lbl = new_label(cint); Yap_emit(label_op, lbl, Zero, cint); cs = emit_cswitch(n, FAILCODE, cint); for (i = 0; i < n; i++) { AtomSwiEntry *ics; ClauseDef *max = min; ics = fetch_centry(cs, min->Tag, i, n); ics->Tag = min->Tag; while (max != grp->LastClause && (max + 1)->Tag == min->Tag) max++; if (min != max) { if (sreg != NULL) { if (ap->PredFlags & LogUpdatePredFlag && max > min) { ics->u_a.Label = suspend_indexing(min, max, ap, cint); } else { ics->u_a.Label = do_compound_index( min, max, sreg, cint, compound_term, arity, argno, nxtlbl, first, last_arg, clleft, top, TRUE); } } else if (ap->PredFlags & LogUpdatePredFlag) { ics->u_a.Label = suspend_indexing(min, max, cint->CurrentPred, cint); } else { ics->u_a.Label = do_index(min, max, cint, argno + 1, nxtlbl, first, clleft, top); } } else { ics->u_a.Label = do_index(min, max, cint, argno + 1, nxtlbl, first, clleft, top); } grp->FirstClause = min = max + 1; } return lbl; } static void do_blobs(GroupDef *grp, Term t, struct intermediates *cint, UInt argno, int first, UInt nxtlbl, int clleft, CELL *top) { COUNT n; ClauseDef *min = grp->FirstClause; COUNT i; /* generate a switch */ AtomSwiEntry *cs; PredEntry *ap = cint->CurrentPred; n = count_blobs(grp); cs = emit_cswitch(n, (yamop *)nxtlbl, cint); for (i = 0; i < n; i++) { AtomSwiEntry *ics; ClauseDef *max = min; ics = fetch_centry(cs, min->Tag, i, n); ics->Tag = min->Tag; while (max != grp->LastClause && (max + 1)->Tag == min->Tag) max++; if (min != max && (ap->PredFlags & LogUpdatePredFlag)) { ics->u_a.Label = suspend_indexing(min, max, ap, cint); } else { ics->u_a.Label = do_index(min, max, cint, argno + 1, nxtlbl, first, clleft, top); } grp->FirstClause = min = max + 1; } } static UInt do_funcs(GroupDef *grp, Term t, struct intermediates *cint, UInt argno, int first, int last_arg, UInt nxtlbl, int clleft, CELL *top) { COUNT n = count_funcs(grp); ClauseDef *min = grp->FirstClause; COUNT i; FuncSwiEntry *fs; UInt lbl; if (min > grp->LastClause || n == 0) { /* no clauses, just skip */ return nxtlbl; } lbl = new_label(cint); Yap_emit(label_op, lbl, Zero, cint); /* generate a switch */ fs = emit_fswitch(n, FAILCODE, cint); for (i = 0; i < n; i++) { Functor f = (Functor)RepAppl(min->Tag); FuncSwiEntry *ifs; ClauseDef *max = min; ifs = fetch_fentry(fs, f, i, n); ifs->Tag = f; while (max != grp->LastClause && (max + 1)->Tag == min->Tag) max++; /* delay non-trivial indexing if (min != max && !IsExtensionFunctor(f)) { ifs->y_u.Label = suspend_indexing(min, max, ap, cint); } else */ if (IsExtensionFunctor(f)) { if (f == FunctorDBRef) ifs->u_f.Label = do_dbref_index(min, max, t, cint, argno, nxtlbl, first, clleft, top); else if (f == FunctorLongInt || f == FunctorBigInt) ifs->u_f.Label = do_blob_index(min, max, t, cint, argno, nxtlbl, first, clleft, top, FALSE); else ifs->u_f.Label = do_blob_index(min, max, t, cint, argno, nxtlbl, first, clleft, top, TRUE); } else { CELL *sreg; if (!IsVarTerm(t) && IsApplTerm(t) && FunctorOfTerm(t) == f) { sreg = RepAppl(t) + 1; } else { sreg = NULL; } ifs->u_f.Label = do_compound_index(min, max, sreg, cint, 0, ArityOfFunctor(f), argno, nxtlbl, first, last_arg, clleft, top, TRUE); } grp->FirstClause = min = max + 1; } return lbl; } static UInt do_pair(GroupDef *grp, Term t, struct intermediates *cint, UInt argno, int first, int last_arg, UInt nxtlbl, int clleft, CELL *top) { ClauseDef *min = grp->FirstClause; ClauseDef *max = grp->FirstClause; while (IsPairTerm(max->Tag) && max != grp->LastClause) { max++; } if (!IsPairTerm(max->Tag)) { max--; } if (min > grp->LastClause) { /* no clauses, just skip */ return nxtlbl; } grp->FirstClause = max + 1; if (min == max) { /* single clause, no need to do indexing, but we do know it is a list */ if (cint->CurrentPred->PredFlags & LogUpdatePredFlag) { return (UInt)(min->Code); } else { return (UInt)(min->CurrentCode); } } if (min != max && !IsPairTerm(t)) { return suspend_indexing(min, max, cint->CurrentPred, cint); } return do_compound_index(min, max, (IsPairTerm(t) ? RepPair(t) : NULL), cint, 0, 2, argno, nxtlbl, first, last_arg, clleft, top, TRUE); } static void group_prologue(int compound_term, UInt argno, int first, struct intermediates *cint) { if (compound_term) { Yap_emit(cache_sub_arg_op, compound_term - 1, compound_term - 1, cint); } else { if (!first || argno != 1) { Yap_emit(cache_arg_op, argno, argno, cint); } } } /* make sure that we can handle failure correctly */ static void emit_protection_choicepoint(int first, int clleft, UInt nxtlbl, struct intermediates *cint) { if (first) { if (clleft) { if (cint->CurrentPred->PredFlags & LogUpdatePredFlag) { UInt labl = new_label(cint); Yap_emit_4ops(enter_lu_op, labl, labl, 0, Zero, cint); Yap_emit(label_op, labl, Zero, cint); } Yap_emit(tryme_op, nxtlbl, (clleft << 1), cint); } } else { /* !first */ if (clleft) { Yap_emit(retryme_op, nxtlbl, (clleft << 1), cint); } else { Yap_emit(trustme_op, 0, 0, cint); } } } static ClauseDef *cls_move(ClauseDef *min, PredEntry *ap, ClauseDef *max, int compound_term, UInt argno, int last_arg) { ClauseDef *cl = min; cl = min; if (compound_term) { while (cl <= max) { skip_to_arg(cl, ap, compound_term, last_arg); cl++; } } else { while (cl <= max) { if (cl->Tag == (_var + 1) * sizeof(CELL)) { ClauseDef *cli = cl; while (cli < max) { clcpy(cli, cli + 1); cli++; } max--; } else { move_next(cl, argno); } cl++; } } return max; } static void purge_pvar(GroupDef *group) { ClauseDef *max = group->LastClause; ClauseDef *cl = group->FirstClause; while (cl <= max) { if (cl->Tag == (_var + 1) * sizeof(CELL)) { ClauseDef *cli = cl; while (cli < max) { clcpy(cli, cli + 1); cli++; } group->VarClauses--; max--; } cl++; } group->LastClause = max; } static UInt *do_nonvar_group(GroupDef *grp, Term t, UInt compound_term, CELL *sreg, UInt arity, UInt labl, struct intermediates *cint, UInt argno, int first, int last_arg, UInt nxtlbl, int clleft, CELL *top) { TypeSwitch *type_sw; PredEntry *ap = cint->CurrentPred; /* move cl pointer */ if (grp->AtomClauses + grp->PairClauses + grp->StructClauses > 1) { Yap_emit(label_op, labl, Zero, cint); if (argno == 1 && !compound_term) { emit_protection_choicepoint(first, clleft, nxtlbl, cint); } group_prologue(compound_term, argno, first, cint); if (grp->LastClause < grp->FirstClause) { /* only tests */ return NULL; } type_sw = emit_type_switch(switch_on_type_op, cint); /* have these first so that we will have something initialized here */ type_sw->ConstEntry = type_sw->FuncEntry = type_sw->PairEntry = type_sw->VarEntry = nxtlbl; type_sw->VarEntry = do_var_entries(grp, t, cint, argno, first, clleft, nxtlbl); grp->LastClause = cls_move(grp->FirstClause, ap, grp->LastClause, compound_term, argno, last_arg); sort_group(grp, top, cint); while (grp->FirstClause <= grp->LastClause) { if (IsAtomOrIntTerm(grp->FirstClause->Tag)) { type_sw->ConstEntry = do_consts(grp, t, cint, compound_term, sreg, arity, last_arg, argno, first, nxtlbl, clleft, top); } else if (IsApplTerm(grp->FirstClause->Tag)) { type_sw->FuncEntry = do_funcs(grp, t, cint, argno, first, last_arg, nxtlbl, clleft, top); } else { type_sw->PairEntry = do_pair(grp, t, cint, argno, first, last_arg, nxtlbl, clleft, top); } } return &(type_sw->VarEntry); } else { Yap_emit(label_op, labl, Zero, cint); do_var_group(grp, cint, TRUE, first, clleft, nxtlbl, ap->ArityOfPE + 1); return NULL; } } static UInt do_optims(GroupDef *group, int ngroups, UInt fail_l, ClauseDef *min, struct intermediates *cint) { if (ngroups == 2 && group[0].FirstClause == group[0].LastClause && group[0].AtomClauses == 1 && group[1].VarClauses == 1) { CELL *sp; UInt labl; labl = new_label(cint); sp = Yap_emit_extra_size(if_not_op, Zero, 4 * CellSize, cint); sp[0] = (CELL)(group[0].FirstClause->Tag); sp[1] = (CELL)(group[1].FirstClause->Code); sp[2] = do_var_clauses(group[0].FirstClause, group[1].LastClause, FALSE, cint, TRUE, 0, (CELL)FAILCODE, cint->CurrentPred->ArityOfPE + 1); sp[3] = do_var_clauses(min, group[1].LastClause, FALSE, cint, TRUE, 0, (CELL)FAILCODE, cint->CurrentPred->ArityOfPE + 1); return labl; } return fail_l; } static int cls_info(ClauseDef *min, ClauseDef *max, UInt argno) { ClauseDef *cl = min; int found_pvar = FALSE; while (cl <= max) { add_info(cl, argno); if (cl->Tag == (_var + 1) * sizeof(CELL)) { found_pvar = TRUE; } /* if (IsVarTerm(cl->Tag)) cl->Tag = (CELL)NULL; */ cl++; } return found_pvar; } static int cls_head_info(ClauseDef *min, ClauseDef *max, UInt argno, int in_idb) { ClauseDef *cl = min; if (in_idb) { if (argno != 2) { while (cl <= max) { cl->Tag = (CELL)NULL; cl++; } } else { while (cl <= max) { LogUpdClause *lcl = ClauseCodeToLogUpdClause(cl->CurrentCode); Term t = lcl->lusl.ClSource->Entry; if (IsVarTerm(t)) { cl->Tag = (CELL)NULL; } else if (IsApplTerm(t)) { CELL *pt = RepAppl(t); cl->Tag = AbsAppl((CELL *)pt[0]); if (IsExtensionFunctor(FunctorOfTerm(t))) { cl->ucd.t_ptr = t; } else { cl->ucd.c_sreg = pt; } } else if (IsPairTerm(t)) { CELL *pt = RepPair(t); cl->Tag = AbsPair(NULL); cl->ucd.c_sreg = pt - 1; } else { cl->Tag = t; } cl++; } } } else { while (cl <= max) { add_info(cl, argno); /* if (IsVarTerm(cl->Tag)) cl->Tag = (CELL)NULL; */ cl++; } } return FALSE; } static UInt do_index(ClauseDef *min, ClauseDef *max, struct intermediates *cint, UInt argno, UInt fail_l, int first, int clleft, CELL *top) { CACHE_REGS UInt ngroups, found_pvar = FALSE; UInt i = 0; GroupDef *group = (GroupDef *)top; UInt labl, labl0, lablx; Term t; /* remember how we entered here */ UInt argno0 = argno; PredEntry *ap = cint->CurrentPred; yamop *eblk = cint->expand_block; if (min == max) { /* base case, just commit to the current code */ return emit_single_switch_case(min, cint, first, clleft, fail_l); } if ((argno > 1 && indexingMode() == TermSingle && ap->PredFlags & LogUpdatePredFlag) || indexingMode() == TermOff || ap->ArityOfPE < argno) { return do_var_clauses(min, max, FALSE, cint, first, clleft, fail_l, ap->ArityOfPE + 1); } t = Deref(XREGS[argno]); if (ap->PredFlags & LogUpdatePredFlag) { found_pvar = cls_head_info(min, max, argno, (ap->ModuleOfPred == IDB_MODULE)); } else { found_pvar = cls_info(min, max, argno); } ngroups = groups_in(min, max, group, cint); if (IsVarTerm(t)) { lablx = new_label(cint); Yap_emit(label_op, lablx, Zero, cint); while (IsVarTerm(t)) { if (ngroups > 1 || !group->VarClauses) { UInt susp_lab = suspend_indexing(min, max, ap, cint); if (!cint->expand_block) { cint->expand_block = (yamop *)susp_lab; } Yap_emit(jump_nv_op, susp_lab, argno, cint); } if (argno == ap->ArityOfPE || (indexingMode() == TermSingle && ap->PredFlags & LogUpdatePredFlag)) { do_var_clauses(min, max, FALSE, cint, first, clleft, fail_l, argno0); cint->expand_block = eblk; return lablx; } argno++; t = Deref(XREGS[argno]); if (ap->PredFlags & LogUpdatePredFlag) { found_pvar = cls_head_info(min, max, argno, (ap->ModuleOfPred == IDB_MODULE)); } else { found_pvar = cls_info(min, max, argno); } ngroups = groups_in(min, max, group, cint); } labl0 = labl = new_label(cint); } else { lablx = labl0 = labl = new_label(cint); } cint->expand_block = eblk; top = (CELL *)(group + ngroups); if (argno > 1) { /* don't try being smart for other arguments than the first */ if (ngroups > 1 || group->VarClauses != 0 || found_pvar) { if (ap->ArityOfPE == argno) { return do_var_clauses(min, max, FALSE, cint, first, clleft, fail_l, ap->ArityOfPE + 1); } else { return do_index(min, max, cint, argno + 1, fail_l, first, clleft, top); } } else { ClauseDef *cl = min; /* need to reset the code pointer, otherwise I could be in the middle of a compound term. */ while (cl <= max) { cl->CurrentCode = cl->Code; cl++; } } } else { UInt special_options; if ((ap->PredFlags & LogUpdatePredFlag) && ngroups > 1) { if (ngroups > 1) { group[0].VarClauses = ap->cs.p_code.NOfClauses; group[0].AtomClauses = group[0].PairClauses = group[0].StructClauses = group[0].TestClauses = 0; group[0].LastClause = group[ngroups - 1].LastClause; ngroups = 1; } } else if ((special_options = do_optims(group, ngroups, fail_l, min, cint)) != fail_l) { return special_options; } if (ngroups == 1 && group->VarClauses && !found_pvar) { return do_index(min, max, cint, argno + 1, fail_l, first, clleft, top); } else if (found_pvar || (ap->PredFlags & LogUpdatePredFlag && group[0].VarClauses)) { /* make sure we know where to suspend */ Yap_emit(label_op, labl0, Zero, cint); labl = new_label(cint); Yap_emit(jump_v_op, suspend_indexing(min, max, ap, cint), Zero, cint); } } for (i = 0; i < ngroups; i++) { UInt nextlbl; int left_clauses = clleft + (max - group->LastClause); /* a group may end up not having clauses*/ if (i < ngroups - 1) { nextlbl = new_label(cint); } else { nextlbl = fail_l; } if (found_pvar && argno == 1) { purge_pvar(group); } if (group->FirstClause == group->LastClause && first && left_clauses == 0) { Yap_emit(jumpi_op, (CELL)(group->FirstClause->Code), Zero, cint); } else { if (group->VarClauses) { Yap_emit(label_op, labl, Zero, cint); do_var_group(group, cint, argno == 1, first, left_clauses, nextlbl, ap->ArityOfPE + 1); } else { do_nonvar_group(group, t, 0, NULL, 0, labl, cint, argno, first, TRUE, nextlbl, left_clauses, top); } } first = FALSE; group++; labl = nextlbl; } return lablx; } static ClauseDef *copy_clauses(ClauseDef *max0, ClauseDef *min0, CELL *top, struct intermediates *cint) { CACHE_REGS UInt sz = ((max0 + 1) - min0) * sizeof(ClauseDef); if ((char *)top + sz >= LOCAL_TrailTop - 4096) { LOCAL_Error_Size = sz; /* grow stack */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 4); } memmove((void *)top, (void *)min0, sz); return (ClauseDef *)top; } /* make sure that it is worth it to generate indexing code at that point */ static int several_tags(ClauseDef *min, ClauseDef *max) { CELL tag = min->Tag; while (min < max) { min++; if (!IsAtomOrIntTerm(min->Tag) || min->Tag != tag) return TRUE; } return FALSE; } /* execute an index inside a structure */ static UInt do_compound_index(ClauseDef *min0, ClauseDef *max0, Term *sreg, struct intermediates *cint, UInt i, UInt arity, UInt argno, UInt fail_l, int first, int last_arg, int clleft, CELL *top, int done_work) { UInt ret_lab = 0, *newlabp; CELL *top0 = top; ClauseDef *min, *max; PredEntry *ap = cint->CurrentPred; int found_index = FALSE; pred_flags_t lu_pred = ap->PredFlags & LogUpdatePredFlag; UInt old_last_depth, old_last_depth_size; newlabp = &ret_lab; if (min0 == max0) { /* base case, just commit to the current code */ return emit_single_switch_case(min0, cint, first, clleft, fail_l); } if ((indexingMode() == TermSingle && ap->PredFlags & LogUpdatePredFlag) || (indexingDepth() && cint->term_depth - cint->last_index_new_depth > indexingDepth())) { *newlabp = do_var_clauses(min0, max0, FALSE, cint, first, clleft, fail_l, ap->ArityOfPE + 1); return ret_lab; } if (sreg == NULL) { return suspend_indexing(min0, max0, ap, cint); } cint->term_depth++; old_last_depth = cint->last_index_new_depth; old_last_depth_size = cint->last_depth_size; if (cint->last_depth_size != max0 - min0) { cint->last_index_new_depth = cint->term_depth; cint->last_depth_size = max0 - min0; } while (i < arity && !found_index) { ClauseDef *cl; GroupDef *group; UInt ngroups; int isvt = IsVarTerm(Deref(sreg[i])); min = copy_clauses(max0, min0, top, cint); max = min + (max0 - min0); top = (CELL *)(max + 1); cl = min; /* search for a subargument */ while (cl <= max) { add_arg_info(cl, ap, i + 1); cl++; } group = (GroupDef *)top; ngroups = groups_in(min, max, group, cint); if (ngroups == 1 && group->VarClauses == 0 && (i < 8 || several_tags(min, max))) { /* ok, we are doing a sub-argument */ /* process group */ found_index = TRUE; ret_lab = new_label(cint); top = (CELL *)(group + 1); if (do_nonvar_group(group, (sreg == NULL ? 0L : Deref(sreg[i])), i + 1, (isvt ? NULL : sreg), arity, *newlabp, cint, argno, first, (last_arg && i + 1 == arity), fail_l, clleft, top) == NULL) { top = top0; break; } } top = top0; i++; } if (!found_index) { if (!lu_pred || !done_work) *newlabp = do_index(min0, max0, cint, argno + 1, fail_l, first, clleft, top); else *newlabp = suspend_indexing(min0, max0, ap, cint); } cint->last_index_new_depth = old_last_depth; cint->last_depth_size = old_last_depth_size; cint->term_depth--; return ret_lab; } static UInt do_dbref_index(ClauseDef *min, ClauseDef *max, Term t, struct intermediates *cint, UInt argno, UInt fail_l, int first, int clleft, CELL *top) { UInt ngroups; GroupDef *group; ClauseDef *cl = min; group = (GroupDef *)top; cl = min; while (cl <= max) { cl->Tag = cl->ucd.t_ptr; cl++; } ngroups = groups_in(min, max, group, cint); if (ngroups > 1 || group->VarClauses) { return do_index(min, max, cint, argno + 1, fail_l, first, clleft, top); } else { int labl = new_label(cint); Yap_emit(label_op, labl, Zero, cint); Yap_emit(index_dbref_op, Zero, Zero, cint); sort_group(group, (CELL *)(group + 1), cint); do_blobs(group, t, cint, argno, first, fail_l, clleft, (CELL *)(group + 1)); return labl; } } static UInt do_blob_index(ClauseDef *min, ClauseDef *max, Term t, struct intermediates *cint, UInt argno, UInt fail_l, int first, int clleft, CELL *top, int blob) { UInt ngroups; GroupDef *group; ClauseDef *cl = min; group = (GroupDef *)top; cl = min; while (cl <= max) { if (cl->ucd.t_ptr == (CELL)NULL) { /* check whether it is a builtin */ cl->Tag = Zero; } else if (blob) { cl->Tag = Yap_Double_key(cl->ucd.t_ptr); } else { cl->Tag = Yap_Int_key(cl->ucd.t_ptr); } cl++; } ngroups = groups_in(min, max, group, cint); if (ngroups > 1 || group->VarClauses) { return do_index(min, max, cint, argno + 1, fail_l, first, clleft, top); } else { int labl = new_label(cint); Yap_emit(label_op, labl, Zero, cint); if (blob) Yap_emit(index_blob_op, Zero, Zero, cint); else Yap_emit(index_long_op, Zero, Zero, cint); sort_group(group, (CELL *)(group + 1), cint); do_blobs(group, t, cint, argno, first, fail_l, clleft, (CELL *)(group + 1)); return labl; } } static void init_clauses(ClauseDef *cl, PredEntry *ap) { if (ap->PredFlags & MegaClausePredFlag) { MegaClause *mcl = ClauseCodeToMegaClause(ap->cs.p_code.FirstClause); UInt nclauses = mcl->ClPred->cs.p_code.NOfClauses; yamop *end = (yamop *)((char *)mcl->ClCode + nclauses * mcl->ClItemSize); yamop *cd = mcl->ClCode; while (cd < end) { cl->Code = cl->CurrentCode = cd; cd = (yamop *)((char *)cd + mcl->ClItemSize); cl++; } } else { StaticClause *scl; scl = ClauseCodeToStaticClause(ap->cs.p_code.FirstClause); do { cl->Code = cl->CurrentCode = scl->ClCode; cl++; if (scl->ClCode == ap->cs.p_code.LastClause) return; scl = scl->ClNext; } while (TRUE); } } static void init_log_upd_clauses(ClauseDef *cl, PredEntry *ap) { LogUpdClause *lcl = ClauseCodeToLogUpdClause(ap->cs.p_code.FirstClause); do { cl->Code = cl->CurrentCode = lcl->ClCode; cl++; lcl = lcl->ClNext; } while (lcl != NULL); } static UInt compile_index(struct intermediates *cint) { CACHE_REGS PredEntry *ap = cint->CurrentPred; int NClauses = ap->cs.p_code.NOfClauses; CELL *top = (CELL *)TR; UInt res; /* only global variable I use directly */ cint->i_labelno = 1; LOCAL_Error_Size = 0; #if USE_SYSTEM_MALLOC if (!cint->cls) { cint->cls = (ClauseDef *)Yap_AllocCodeSpace(NClauses * sizeof(ClauseDef)); if (!cint->cls) { /* tell how much space we need */ LOCAL_Error_Size += NClauses * sizeof(ClauseDef); /* grow stack */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 2); } } cint->freep = (char *)HR; #else /* reserve double the space for compiler */ cint->cls = (ClauseDef *)HR; if (cint->cls + 2 * NClauses > (ClauseDef *)(ASP - 4096)) { /* tell how much space we need */ LOCAL_Error_Size += NClauses * sizeof(ClauseDef); /* grow stack */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 3); } cint->freep = (char *)(cint->cls + NClauses); #endif if (ap->PredFlags & LogUpdatePredFlag) { /* throw away a label */ new_label(cint); init_log_upd_clauses(cint->cls, ap); } else if (ap->PredFlags & UDIPredFlag) { UInt lbl = new_label(cint); Yap_emit(user_switch_op, Unsigned(ap), Unsigned(&(ap->cs.p_code.ExpandCode)), cint); return lbl; } else { /* prepare basic data structures */ init_clauses(cint->cls, ap); } res = do_index(cint->cls, cint->cls + (NClauses - 1), cint, 1, (UInt)FAILCODE, TRUE, 0, top); return res; } static void CleanCls(struct intermediates *cint) { #if USE_SYSTEM_MALLOC if (cint->cls) { Yap_FreeCodeSpace((ADDR)cint->cls); } #endif cint->cls = NULL; } yamop *Yap_PredIsIndexable(PredEntry *ap, UInt NSlots, yamop *next_pc) { CACHE_REGS yamop *indx_out; int setjres; struct intermediates cint; cint.CurrentPred = ap; cint.code_addr = NULL; cint.blks = NULL; cint.cls = NULL; LOCAL_Error_Size = 0; if (ap->cs.p_code.NOfClauses < 2) return NULL; if ((setjres = sigsetjmp(cint.CompilerBotch, 0)) == 3) { restore_machine_regs(); recover_from_failed_susp_on_cls(&cint, 0); if (!Yap_gcl(LOCAL_Error_Size, ap->ArityOfPE + NSlots, ENV, next_pc)) { CleanCls(&cint); Yap_Error(RESOURCE_ERROR_STACK, TermNil, LOCAL_ErrorMessage); return NULL; } } else if (setjres == 2) { restore_machine_regs(); LOCAL_Error_Size = recover_from_failed_susp_on_cls(&cint, LOCAL_Error_Size); if (!Yap_growheap(FALSE, LOCAL_Error_Size, NULL)) { CleanCls(&cint); Yap_Error(RESOURCE_ERROR_HEAP, TermNil, LOCAL_ErrorMessage); return NULL; } } else if (setjres == 4) { restore_machine_regs(); recover_from_failed_susp_on_cls(&cint, 0); if (!Yap_growtrail(LOCAL_Error_Size, FALSE)) { CleanCls(&cint); Yap_Error(RESOURCE_ERROR_TRAIL, TermNil, LOCAL_ErrorMessage); return NULL; } } else if (setjres != 0) { restore_machine_regs(); recover_from_failed_susp_on_cls(&cint, 0); if (!Yap_growheap(FALSE, LOCAL_Error_Size, NULL)) { Yap_Error(RESOURCE_ERROR_HEAP, TermNil, LOCAL_ErrorMessage); CleanCls(&cint); return NULL; } } restart_index: Yap_BuildMegaClause(ap); cint.CodeStart = cint.BlobsStart = cint.cpc = cint.icpc = NULL; cint.expand_block = NULL; cint.label_offset = NULL; LOCAL_ErrorMessage = NULL; cint.term_depth = cint.last_index_new_depth = cint.last_depth_size = 0L; if (compile_index(&cint) == (UInt)FAILCODE) { Yap_ReleaseCMem(&cint); CleanCls(&cint); return NULL; } #if DEBUG if (GLOBAL_Option['i' - 'a' + 1]) { Yap_ShowCode(&cint); } #endif /* globals for assembler */ LOCAL_IPredArity = ap->ArityOfPE; if (cint.CodeStart) { if ((indx_out = Yap_assemble(ASSEMBLING_INDEX, TermNil, ap, FALSE, &cint, cint.i_labelno + 1)) == NULL) { if (!Yap_growheap(FALSE, LOCAL_Error_Size, NULL)) { Yap_ReleaseCMem(&cint); CleanCls(&cint); Yap_Error(RESOURCE_ERROR_HEAP, TermNil, LOCAL_ErrorMessage); return NULL; } goto restart_index; } } else { Yap_ReleaseCMem(&cint); CleanCls(&cint); return NULL; } Yap_ReleaseCMem(&cint); CleanCls(&cint); if (ap->PredFlags & LogUpdatePredFlag) { LogUpdIndex *cl = ClauseCodeToLogUpdIndex(indx_out); cl->ClFlags |= SwitchRootMask; } return (indx_out); } static istack_entry *push_stack(istack_entry *sp, Int arg, Term Tag, Term extra, struct intermediates *cint) { CACHE_REGS if (sp + 1 > (istack_entry *)LOCAL_TrailTop) { save_machine_regs(); siglongjmp(cint->CompilerBotch, 4); } sp->pos = arg; sp->val = Tag; sp->extra = extra; sp++; sp->pos = 0; return sp; } static istack_entry *install_clause(ClauseDef *cls, PredEntry *ap, istack_entry *stack) { istack_entry *sp = stack; while (sp->pos) { if ((Int)(sp->pos) > 0) { add_info(cls, sp->pos); } else if (sp->pos) { UInt argno = -sp->pos; add_arg_info(cls, ap, argno); } /* if we are not talking about a variable */ if (cls->Tag != sp->val) { if (sp->val == 0L) { sp++; } break; } else { if (IsApplTerm(cls->Tag)) { Functor f = (Functor)RepAppl(cls->Tag); if (IsExtensionFunctor(f)) { if (f == FunctorDBRef) { if (cls->ucd.t_ptr != sp->extra) break; } else if (f == FunctorDouble) { if (cls->ucd.t_ptr && Yap_Double_key(sp->extra) != Yap_Double_key(cls->ucd.t_ptr)) break; } else if (f == FunctorString) { if (cls->ucd.t_ptr && Yap_String_key(sp->extra) != Yap_String_key(cls->ucd.t_ptr)) break; } else { if (cls->ucd.t_ptr && Yap_Int_key(sp->extra) != Yap_Int_key(cls->ucd.t_ptr)) break; } } } if ((Int)(sp->pos) > 0) { move_next(cls, sp->pos); } else if (sp->pos) { UInt argno = -sp->pos; skip_to_arg(cls, ap, argno, FALSE); } } sp++; } return sp; } static ClauseDef *install_clauses(ClauseDef *cls, PredEntry *ap, istack_entry *stack, yamop *beg, yamop *end) { istack_entry *sp = stack; if (ap->PredFlags & MegaClausePredFlag) { MegaClause *mcl = ClauseCodeToMegaClause(beg); UInt nclauses = mcl->ClPred->cs.p_code.NOfClauses; yamop *end = (yamop *)((char *)mcl->ClCode + nclauses * mcl->ClItemSize); yamop *cd = mcl->ClCode; if (stack[0].pos == 0) { while (TRUE) { cls->Code = cls->CurrentCode = cd; cls->Tag = 0; cls++; cd = (yamop *)((char *)cd + mcl->ClItemSize); if (cd == end) { return cls - 1; } } } while (TRUE) { cls->Code = cls->CurrentCode = cd; sp = install_clause(cls, ap, stack); /* we reached a matching clause */ if (!sp->pos && (sp[-1].val == 0L || cls->Tag == sp[-1].val)) { cls++; } cd = (yamop *)((char *)cd + mcl->ClItemSize); if (cd == end) { return cls - 1; } } } else { StaticClause *cl = ClauseCodeToStaticClause(beg); if (stack[0].pos == 0) { while (TRUE) { cls->Code = cls->CurrentCode = cl->ClCode; cls->Tag = 0; cls++; if (cl->ClCode == end) { return cls - 1; } cl = cl->ClNext; } } while (TRUE) { cls->Code = cls->CurrentCode = cl->ClCode; sp = install_clause(cls, ap, stack); /* we reached a matching clause */ if (!sp->pos && (sp[-1].val == 0L || cls->Tag == sp[-1].val)) { cls++; } if (cl->ClCode == end) { return cls - 1; } cl = cl->ClNext; } } } static ClauseDef *install_clauseseq(ClauseDef *cls, PredEntry *ap, istack_entry *stack, yamop **beg, yamop **end) { istack_entry *sp = stack; if (stack[0].pos == 0) { while (TRUE) { if (*beg) { cls->Code = cls->CurrentCode = *beg; cls->Tag = 0; cls++; } beg++; if (beg == end) { return cls - 1; } } } while (TRUE) { if (*beg) { cls->Code = cls->CurrentCode = *beg; sp = install_clause(cls, ap, stack); /* we reached a matching clause */ if (!sp->pos && (sp[-1].val == 0L || cls->Tag == sp[-1].val)) { cls++; } } beg++; if (beg == end) { return cls - 1; } } } static void reinstall_clauses(ClauseDef *cls, ClauseDef *end, PredEntry *ap, istack_entry *stack) { do { cls->CurrentCode = cls->Code; install_clause(cls, ap, stack); } while (cls++ != end); } static istack_entry *install_log_upd_clause(ClauseDef *cls, PredEntry *ap, istack_entry *stack) { istack_entry *sp = stack; while (sp->pos) { if ((Int)(sp->pos) > 0) { add_head_info(cls, sp->pos); } else if (sp->pos) { UInt argno = -sp->pos; add_arg_info(cls, ap, argno); } /* if we are not talking about a variable */ if (cls->Tag != sp->val) { if (sp->val == 0L) { sp++; } break; } else { if (IsApplTerm(cls->Tag)) { Functor f = (Functor)RepAppl(cls->Tag); if (IsExtensionFunctor(f)) { if (f == FunctorDBRef) { if (cls->ucd.t_ptr != sp->extra) break; } else if (f == FunctorDouble) { if (cls->ucd.t_ptr && Yap_Double_key(sp->extra) != Yap_Double_key(cls->ucd.t_ptr)) break; } else { if (cls->ucd.t_ptr && Yap_Int_key(sp->extra) != Yap_Int_key(cls->ucd.t_ptr)) break; } } } if ((Int)(sp->pos) > 0) { move_next(cls, sp->pos); } else if (sp->pos) { UInt argno = -sp->pos; skip_to_arg(cls, ap, argno, FALSE); } } sp++; } return sp; } static ClauseDef *install_log_upd_clauses(ClauseDef *cls, PredEntry *ap, istack_entry *stack, yamop *beg, yamop *end) { istack_entry *sp = stack; if (stack[0].pos == 0) { while (TRUE) { cls->Code = cls->CurrentCode = beg; cls->Tag = 0; cls++; if (beg == end || beg == NULL) { return cls - 1; } beg = ClauseCodeToLogUpdClause(beg)->ClNext->ClCode; } } while (TRUE) { cls->Code = cls->CurrentCode = beg; sp = install_log_upd_clause(cls, ap, stack); /* we reached a matching clause */ if (!sp->pos && (sp[-1].val == 0L || cls->Tag == sp[-1].val)) { cls++; } if (beg == end || beg == NULL) { return cls - 1; } beg = ClauseCodeToLogUpdClause(beg)->ClNext->ClCode; } } static ClauseDef *install_log_upd_clauseseq(ClauseDef *cls, PredEntry *ap, istack_entry *stack, yamop **beg, yamop **end) { istack_entry *sp = stack; if (stack[0].pos == 0) { while (TRUE) { if (beg) { cls->Code = cls->CurrentCode = *beg; cls->Tag = 0; cls++; } beg++; if (beg == end) { return cls - 1; } } } while (TRUE) { if (*beg) { cls->Code = cls->CurrentCode = *beg; sp = install_log_upd_clause(cls, ap, stack); /* we reached a matching clause */ if (!sp->pos && (sp[-1].val == 0L || cls->Tag == sp[-1].val)) { cls++; } } beg++; if (beg == end) { return cls - 1; } } } static void reinstall_log_upd_clauses(ClauseDef *cls, ClauseDef *end, PredEntry *ap, istack_entry *stack) { do { cls->CurrentCode = cls->Code; install_log_upd_clause(cls, ap, stack); } while (cls++ != end); } #if PRECOMPUTE_REGADDRESS #define arg_from_x(I) (((CELL *)(I)) - XREGS) #else #define arg_from_x(I) (I) #endif /* ALIGN_LONGS */ static AtomSwiEntry *lookup_c(Term t, yamop *tab, COUNT entries) { AtomSwiEntry *cebase = (AtomSwiEntry *)tab; while (cebase->Tag != t) { entries--; cebase++; if (entries == 0) return cebase; } return cebase; } static FuncSwiEntry *lookup_f(Functor f, yamop *tab, COUNT entries) { FuncSwiEntry *febase = (FuncSwiEntry *)tab; while (febase->Tag != f) { entries--; febase++; if (entries == 0) return febase; } return febase; } static COUNT count_clauses_left(yamop *cl, PredEntry *ap) { if (ap->PredFlags & LogUpdatePredFlag) { LogUpdClause *c = ClauseCodeToLogUpdClause(cl); COUNT i = 0; while (c != NULL) { i++; c = c->ClNext; } return i; } else if (ap->PredFlags & MegaClausePredFlag) { MegaClause *mcl = ClauseCodeToMegaClause(ap->cs.p_code.FirstClause); UInt ncls = mcl->ClPred->cs.p_code.NOfClauses; return (ncls - 1) - ((char *)cl - (char *)mcl->ClCode) / mcl->ClItemSize; } else { yamop *last = ap->cs.p_code.LastClause; StaticClause *c; COUNT i = 1; c = ClauseCodeToStaticClause(cl); while (c->ClCode != last) { i++; c = c->ClNext; } return i; } } /* We have jumped across indexing code. Check if we jumped within the current indexing block, if we moved back to a parent, or if we jumped to a child. */ static ClausePointer index_jmp(ClausePointer cur, ClausePointer parent, yamop *ipc, int is_lu, yamop *e_code) { if (cur.lui == NULL || ipc == FAILCODE || ipc == e_code || ipc->opc == Yap_opcode(_expand_clauses)) return cur; if (is_lu) { LogUpdIndex *lcur = cur.lui, *ncur; /* check myself */ if (ipc >= lcur->ClCode && ipc < (yamop *)((CODEADDR)lcur + lcur->ClSize)) return cur; /* check if I am returning back to a parent, eg switch with intermediate node */ if (lcur->ParentIndex) { LogUpdIndex *pcur = lcur->ParentIndex; if (ipc >= pcur->ClCode && ipc < (yamop *)((CODEADDR)pcur + pcur->ClSize)) { cur.lui = pcur; return cur; } } /* maybe I am a new group */ ncur = ClauseCodeToLogUpdIndex(ipc); if (ncur->ParentIndex != lcur) { #if DEBUG fprintf(stderr, "OOPS, bad parent in lu index\n"); #endif cur.lui = NULL; return cur; } cur.lui = ncur; return cur; } else { StaticIndex *scur = parent.si, *ncur; /* check myself */ if (!scur) return cur; if (ipc >= scur->ClCode && ipc < (yamop *)((CODEADDR)scur + scur->ClSize)) return cur; ncur = ClauseCodeToStaticIndex(ipc); if (ncur->ClPred == scur->ClPred) { cur.si = ncur; return cur; } /* if (parent.si != cur.si) { if (parent.si) { StaticIndex *pcur = parent.si; if (ipc >= pcur->ClCode && ipc < (yamop *)((CODEADDR)pcur+pcur->ClSize)) return parent; } } cur.si = ncur; return cur; */ cur.si = NULL; return cur; } } static ClausePointer code_to_indexcl(yamop *ipc, int is_lu) { ClausePointer ret; if (is_lu) ret.lui = ClauseCodeToLogUpdIndex(ipc); else ret.si = ClauseCodeToStaticIndex(ipc); return ret; } /* CALLED by expand when entering sub_arg */ static void increase_expand_depth(yamop *ipc, struct intermediates *cint) { yamop *ncode; cint->term_depth++; if (ipc->opc == Yap_opcode(_switch_on_sub_arg_type) && (ncode = ipc->y_u.sllll.l4)->opc == Yap_opcode(_expand_clauses)) { if (ncode->y_u.sssllp.s2 != cint->last_depth_size) { cint->last_index_new_depth = cint->term_depth; cint->last_depth_size = ncode->y_u.sssllp.s2; } } } static void zero_expand_depth(PredEntry *ap, struct intermediates *cint) { cint->term_depth = cint->last_index_new_depth; cint->last_depth_size = ap->cs.p_code.NOfClauses; } static yamop **expand_index(struct intermediates *cint) { CACHE_REGS /* first clause */ PredEntry *ap = cint->CurrentPred; yamop *first, *last = NULL, *alt = NULL; istack_entry *stack, *sp; ClauseDef *max; int NClauses; /* last clause to experiment with */ yamop *ipc; /* labp should point at the beginning of the sequence */ yamop **labp = NULL; ClausePointer parentcl; Term t = TermNil, *s_reg = NULL; int is_last_arg = TRUE; int argno = 1; int isfirstcl = TRUE; /* this is will be used as a new PC */ CELL *top = (CELL *)TR; UInt arity = 0; UInt lab, fail_l, clleft, i = 0; int is_lu = ap->PredFlags & LogUpdatePredFlag; yamop *e_code = (yamop *)&(ap->cs.p_code.ExpandCode); ipc = ap->cs.p_code.TrueCodeOfPred; first = ap->cs.p_code.FirstClause; NClauses = ap->cs.p_code.NOfClauses; sp = stack = (istack_entry *)top; cint->i_labelno = 1; stack[0].pos = 0; /* try to refine the interval using the indexing code */ cint->term_depth = cint->last_index_new_depth = cint->last_depth_size = 0L; parentcl = code_to_indexcl(ipc, is_lu); while (ipc != NULL) { op_numbers op; op = Yap_op_from_opcode(ipc->opc); switch (op) { case _try_clause: case _retry: /* this clause had no indexing */ if (ap->PredFlags & LogUpdatePredFlag) { first = ClauseCodeToLogUpdClause(ipc->y_u.Otapl.d)->ClNext->ClCode; } else if (ap->PredFlags & MegaClausePredFlag) { MegaClause *mcl = ClauseCodeToMegaClause(ap->cs.p_code.FirstClause); first = (yamop *)((char *)ipc->y_u.Otapl.d) + mcl->ClItemSize; } else { first = ClauseCodeToStaticClause(ipc->y_u.Otapl.d)->ClNext->ClCode; } isfirstcl = FALSE; ipc = NEXTOP(ipc, Otapl); break; #if TABLING case _table_try: case _table_retry: /* this clause had no indexing */ first = ClauseCodeToStaticClause(PREVOP(ipc->y_u.Otapl.d, Otapl)) ->ClNext->ClCode; isfirstcl = FALSE; ipc = NEXTOP(ipc, Otapl); break; #endif /* TABLING */ case _try_clause2: case _try_clause3: case _try_clause4: case _retry2: case _retry3: case _retry4: case _try_in: if (ap->PredFlags & LogUpdatePredFlag) { first = ClauseCodeToLogUpdClause(ipc->y_u.l.l)->ClNext->ClCode; } else if (ap->PredFlags & MegaClausePredFlag) { MegaClause *mcl = ClauseCodeToMegaClause(ap->cs.p_code.FirstClause); first = (yamop *)((char *)ipc->y_u.Otapl.d) + mcl->ClItemSize; } else { first = ClauseCodeToStaticClause(ipc->y_u.l.l)->ClNext->ClCode; } isfirstcl = FALSE; ipc = NEXTOP(ipc, l); break; case _retry_me: #ifdef TABLING case _table_retry_me: #endif isfirstcl = FALSE; case _try_me: #ifdef TABLING case _table_try_me: #endif /* ok, we found the start for an indexing block, but we don't if we are going to operate here or not */ /* if we are to commit here, alt will tell us where */ alt = ipc->y_u.Otapl.d; ipc = NEXTOP(ipc, Otapl); /* start of a group, reset stack */ sp = stack; stack[0].pos = 0; break; case _profiled_trust_me: case _trust_me: case _count_trust_me: #ifdef TABLING case _table_trust_me: #endif /* TABLING */ /* we will commit to this group for sure */ ipc = NEXTOP(ipc, Otapl); alt = NULL; /* start of a group, reset stack */ sp = stack; stack[0].pos = 0; break; case _trust: /* we should never be here */ Yap_Error(SYSTEM_ERROR_COMPILER, TermNil, "found trust in expand_index"); labp = NULL; ipc = NULL; break; /* should we ever be here ? I think not */ case _try_logical: case _retry_logical: case _count_retry_logical: case _profiled_retry_logical: ipc = ipc->y_u.OtaLl.n; break; case _trust_logical: case _count_trust_logical: case _profiled_trust_logical: ipc = ipc->y_u.OtILl.n; break; case _enter_lu_pred: /* no useful info */ ipc = ipc->y_u.Illss.l1; break; case _retry_profiled: case _count_retry: /* no useful info */ ipc = NEXTOP(ipc, l); break; case _jump: /* just skip for now, but should worry about memory management */ ipc = ipc->y_u.l.l; /* I don't know how up I will go */ parentcl.si = NULL; break; case _lock_lu: case _procceed: ipc = NEXTOP(ipc, p); break; case _unlock_lu: ipc = NEXTOP(ipc, e); break; case _jump_if_var: if (IsVarTerm(Deref(ARG1))) { labp = &(ipc->y_u.l.l); ipc = ipc->y_u.l.l; parentcl = index_jmp(parentcl, parentcl, ipc, is_lu, e_code); } else { ipc = NEXTOP(ipc, l); } break; case _jump_if_nonvar: argno = arg_from_x(ipc->y_u.xll.x); t = Deref(XREGS[argno]); i = 0; /* expand_index expects to find the new argument */ if (!IsVarTerm(t)) { argno--; labp = &(ipc->y_u.xll.l1); ipc = ipc->y_u.xll.l1; parentcl = index_jmp(parentcl, parentcl, ipc, is_lu, e_code); } else { ipc = NEXTOP(ipc, xll); } break; /* instructions type EC */ /* instructions type e */ case _index_dbref: if (s_reg[-1] != (CELL)FunctorDBREF) { ipc = alt; alt = NULL; break; } t = AbsAppl(s_reg - 1); sp[-1].extra = t; s_reg = NULL; ipc = NEXTOP(ipc, e); break; case _index_blob: if (s_reg[-1] != (CELL)FunctorDouble) { ipc = alt; alt = NULL; break; } t = Yap_DoubleP_key(s_reg); sp[-1].extra = AbsAppl(s_reg - 1); s_reg = NULL; ipc = NEXTOP(ipc, e); break; case _index_long: if (s_reg[-1] != (CELL)FunctorLongInt) { ipc = alt; alt = NULL; break; } t = Yap_IntP_key(s_reg); sp[-1].extra = AbsAppl(s_reg - 1); s_reg = NULL; ipc = NEXTOP(ipc, e); break; case _user_switch: labp = &(ipc->y_u.lp.l); ipc = ipc->y_u.lp.l; break; /* instructions type e */ case _switch_on_type: zero_expand_depth(ap, cint); t = Deref(ARG1); argno = 1; i = 0; if (IsVarTerm(t)) { labp = &(ipc->y_u.llll.l4); ipc = ipc->y_u.llll.l4; } else if (IsPairTerm(t)) { sp = push_stack(sp, 1, AbsPair(NULL), TermNil, cint); s_reg = RepPair(t); labp = &(ipc->y_u.llll.l1); ipc = ipc->y_u.llll.l1; increase_expand_depth(ipc, cint); } else if (IsApplTerm(t)) { sp = push_stack(sp, 1, AbsAppl((CELL *)FunctorOfTerm(t)), TermNil, cint); ipc = ipc->y_u.llll.l3; increase_expand_depth(ipc, cint); } else { sp = push_stack(sp, argno, t, TermNil, cint); ipc = ipc->y_u.llll.l2; } parentcl = index_jmp(parentcl, parentcl, ipc, is_lu, e_code); break; case _switch_list_nl: zero_expand_depth(ap, cint); t = Deref(ARG1); argno = 1; i = 0; if (IsVarTerm(t)) { labp = &(ipc->y_u.ollll.l4); ipc = ipc->y_u.ollll.l4; } else if (IsPairTerm(t)) { s_reg = RepPair(t); labp = &(ipc->y_u.ollll.l1); sp = push_stack(sp, 1, AbsPair(NULL), TermNil, cint); ipc = ipc->y_u.ollll.l1; increase_expand_depth(ipc, cint); } else if (t == TermNil) { sp = push_stack(sp, 1, t, TermNil, cint); ipc = ipc->y_u.ollll.l2; increase_expand_depth(ipc, cint); } else { Term tn; if (IsApplTerm(t)) { tn = AbsAppl((CELL *)FunctorOfTerm(t)); } else { tn = t; } sp = push_stack(sp, argno, tn, TermNil, cint); ipc = ipc->y_u.ollll.l3; } parentcl = index_jmp(parentcl, parentcl, ipc, is_lu, e_code); break; case _switch_on_arg_type: zero_expand_depth(ap, cint); argno = arg_from_x(ipc->y_u.xllll.x); i = 0; t = Deref(XREGS[argno]); if (IsVarTerm(t)) { labp = &(ipc->y_u.xllll.l4); ipc = ipc->y_u.xllll.l4; } else if (IsPairTerm(t)) { s_reg = RepPair(t); sp = push_stack(sp, argno, AbsPair(NULL), TermNil, cint); labp = &(ipc->y_u.xllll.l1); ipc = ipc->y_u.xllll.l1; increase_expand_depth(ipc, cint); } else if (IsApplTerm(t)) { sp = push_stack(sp, argno, AbsAppl((CELL *)FunctorOfTerm(t)), TermNil, cint); ipc = ipc->y_u.xllll.l3; increase_expand_depth(ipc, cint); } else { sp = push_stack(sp, argno, t, TermNil, cint); ipc = ipc->y_u.xllll.l2; } parentcl = index_jmp(parentcl, parentcl, ipc, is_lu, e_code); break; case _switch_on_sub_arg_type: i = ipc->y_u.sllll.s; t = Deref(s_reg[i]); if (i != arity - 1) is_last_arg = FALSE; t = Deref(s_reg[i]); if (IsVarTerm(t)) { labp = &(ipc->y_u.sllll.l4); ipc = ipc->y_u.sllll.l4; i++; } else if (IsPairTerm(t)) { s_reg = RepPair(t); sp = push_stack(sp, -i - 1, AbsPair(NULL), TermNil, cint); labp = &(ipc->y_u.sllll.l1); ipc = ipc->y_u.sllll.l1; i = 0; increase_expand_depth(ipc, cint); } else if (IsApplTerm(t)) { sp = push_stack(sp, -i - 1, AbsAppl((CELL *)FunctorOfTerm(t)), TermNil, cint); ipc = ipc->y_u.sllll.l3; i = 0; increase_expand_depth(ipc, cint); } else { /* We don't push stack here, instead we go over to next argument sp = push_stack(sp, -i-1, t, cint); */ sp = push_stack(sp, -i - 1, t, TermNil, cint); ipc = ipc->y_u.sllll.l2; i++; } parentcl = index_jmp(parentcl, parentcl, ipc, is_lu, e_code); break; case _if_not_then: labp = NULL; ipc = NULL; break; /* instructions type ollll */ case _switch_on_func: case _if_func: case _go_on_func: { FuncSwiEntry *fe; yamop *newpc; Functor f; s_reg = RepAppl(t); f = (Functor)(*s_reg++); if (op == _switch_on_func) { fe = lookup_f_hash(f, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } else { fe = lookup_f(f, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } newpc = fe->u_f.labp; labp = &(fe->u_f.labp); if (newpc == e_code) { /* we found it */ parentcl = code_to_indexcl(ipc->y_u.sssl.l, is_lu); ipc = NULL; } else { ClausePointer npar = code_to_indexcl(ipc->y_u.sssl.l, is_lu); ipc = newpc; parentcl = index_jmp(npar, parentcl, ipc, is_lu, e_code); } } break; case _switch_on_cons: case _if_cons: case _go_on_cons: { AtomSwiEntry *ae; if (op == _switch_on_cons) { ae = lookup_c_hash(t, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } else { ae = lookup_c(t, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } labp = &(ae->u_a.labp); if (ae->u_a.labp == e_code) { /* we found it */ parentcl = code_to_indexcl(ipc->y_u.sssl.l, is_lu); ipc = NULL; } else { ClausePointer npar = code_to_indexcl(ipc->y_u.sssl.l, is_lu); ipc = ae->u_a.labp; parentcl = index_jmp(npar, parentcl, ipc, is_lu, e_code); } } break; case _expand_index: case _expand_clauses: if (alt != NULL && ap->PredFlags & LogUpdatePredFlag) { op_numbers fop = Yap_op_from_opcode(alt->opc); if (fop == _enter_lu_pred) alt = alt->y_u.Illss.l1; } ipc = NULL; break; case _op_fail: ipc = alt; alt = NULL; break; default: if (alt == NULL) { Yap_Error(SYSTEM_ERROR_COMPILER, t, "unexpected instruction %d at expand_index ", op); labp = NULL; ipc = NULL; } else { /* backtrack */ first = alt->y_u.Otapl.d; ipc = alt; alt = NULL; } } } /* if there was an overflow while generating the code, make sure S is still correct */ if (is_lu) { cint->current_cl.lui = parentcl.lui; } else { cint->current_cl.si = parentcl.si; } if (s_reg != NULL) S = s_reg; #ifdef TABLING /* handle tabling hack that insertes a failcode, this really corresponds to not having any more clauses */ if (alt == TRUSTFAILCODE) alt = NULL; #endif if (alt == NULL) { /* oops, we are at last clause */ fail_l = (UInt)FAILCODE; clleft = 0; last = ap->cs.p_code.LastClause; } else { if (ap->PredFlags & LogUpdatePredFlag) { op_numbers op = Yap_op_from_opcode(alt->opc); /* can we be here */ if (op >= _retry2 && op <= _retry4) { last = alt->y_u.l.l; } else { last = alt->y_u.Otapl.d; } } else { op_numbers op = Yap_op_from_opcode(alt->opc); if (op == _retry || op == _trust) { last = alt->y_u.Otapl.d; #ifdef TABLING } else if (op == _table_retry || op == _table_trust) { last = PREVOP(alt->y_u.Otapl.d, Otapl); #endif /* TABLING */ } else if (op >= _retry2 && op <= _retry4) { last = alt->y_u.l.l; } } fail_l = (UInt)alt; clleft = count_clauses_left(last, ap); } if (Yap_op_from_opcode((*labp)->opc) == _expand_clauses) { /* ok, we know how many clauses */ yamop *ipc = *labp; /* check all slots, not just the ones with values */ COUNT nclauses = ipc->y_u.sssllp.s1; yamop **clp = (yamop **)NEXTOP(ipc, sssllp); cint->expand_block = ipc; #if USE_SYSTEM_MALLOC if (!cint->cls) { cint->cls = (ClauseDef *)Yap_AllocCodeSpace(nclauses * sizeof(ClauseDef)); if (!cint->cls) { /* tell how much space we need */ LOCAL_Error_Size += NClauses * sizeof(ClauseDef); /* grow stack */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 2); } } #else cint->cls = (ClauseDef *)HR; if (cint->cls + 2 * nclauses > (ClauseDef *)(ASP - 4096)) { /* tell how much space we need (worst case) */ LOCAL_Error_Size += 2 * NClauses * sizeof(ClauseDef); /* grow stack */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 3); } #endif if (ap->PredFlags & LogUpdatePredFlag) { max = install_log_upd_clauseseq(cint->cls, ap, stack, clp, clp + nclauses); } else { max = install_clauseseq(cint->cls, ap, stack, clp, clp + nclauses); } } else { cint->expand_block = NULL; #if USE_SYSTEM_MALLOC if (!cint->cls) { cint->cls = (ClauseDef *)Yap_AllocCodeSpace(NClauses * sizeof(ClauseDef)); if (!cint->cls) { /* tell how much space we need */ LOCAL_Error_Size += NClauses * sizeof(ClauseDef); /* grow stack */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 2); } } #else cint->cls = (ClauseDef *)HR; if (cint->cls + 2 * NClauses > (ClauseDef *)(ASP - 4096)) { /* tell how much space we need (worst case) */ LOCAL_Error_Size += 2 * NClauses * sizeof(ClauseDef); save_machine_regs(); siglongjmp(cint->CompilerBotch, 3); } #endif if (ap->PredFlags & LogUpdatePredFlag) { max = install_log_upd_clauses(cint->cls, ap, stack, first, last); } else { max = install_clauses(cint->cls, ap, stack, first, last); } #if DEBUG_EXPAND if (ap->PredFlags & LogUpdatePredFlag) { fprintf(stderr, "vsc +"); } else { fprintf(stderr, "vsc "); } fprintf(stderr, " : expanding %d out of %d\n", (max - cls) + 1, NClauses); #endif } /* don't count last clause if you don't have to */ if (alt && max->Code == last) max--; if (max < cint->cls && labp != NULL) { *labp = FAILCODE; return labp; } #if USE_SYSTEM_MALLOC cint->freep = (char *)HR; #else cint->freep = (char *)(max + 1); #endif cint->CodeStart = cint->BlobsStart = cint->cpc = cint->icpc = NULL; if (!IsVarTerm(sp[-1].val) && sp > stack) { if (IsAtomOrIntTerm(sp[-1].val)) { if (s_reg == NULL) { /* we have not yet looked into terms */ lab = do_index(cint->cls, max, cint, argno + 1, fail_l, isfirstcl, clleft, top); } else { UInt arity = 0; if (ap->PredFlags & LogUpdatePredFlag) { reinstall_log_upd_clauses(cint->cls, max, ap, stack); } else { reinstall_clauses(cint->cls, max, ap, stack); } sp--; while (sp > stack) { Term t = sp[-1].val; if (IsApplTerm(t)) { Functor f = (Functor)RepAppl(t); if (!IsExtensionFunctor(f)) { arity = ArityOfFunctor(f); break; } else { sp--; } } else if (IsPairTerm(t)) { arity = 2; break; } else { sp--; } } lab = do_compound_index(cint->cls, max, s_reg, cint, i, arity, argno, fail_l, isfirstcl, is_last_arg, clleft, top, FALSE); } } else if (IsPairTerm(sp[-1].val) && sp > stack) { lab = do_compound_index(cint->cls, max, s_reg, cint, i, 2, argno, fail_l, isfirstcl, is_last_arg, clleft, top, FALSE); } else { Functor f = (Functor)RepAppl(sp[-1].val); /* we are continuing within a compound term */ if (IsExtensionFunctor(f)) { lab = do_index(cint->cls, max, cint, argno + 1, fail_l, isfirstcl, clleft, top); } else { lab = do_compound_index(cint->cls, max, s_reg, cint, i, ArityOfFunctor(f), argno, fail_l, isfirstcl, is_last_arg, clleft, top, FALSE); } } } else { if (argno == ap->ArityOfPE) { lab = do_var_clauses(cint->cls, max, FALSE, cint, isfirstcl, clleft, fail_l, ap->ArityOfPE + 1); } else { lab = do_index(cint->cls, max, cint, argno + 1, fail_l, isfirstcl, clleft, top); } } if (labp && !(lab & 1)) { *labp = (yamop *)lab; /* in case we have a single clause */ } return labp; } static yamop *ExpandIndex(PredEntry *ap, int ExtraArgs, yamop *nextop USES_REGS) { yamop *indx_out, *expand_clauses; yamop **labp; int cb; struct intermediates cint; cint.blks = NULL; cint.cls = NULL; cint.code_addr = NULL; cint.label_offset = NULL; if ((cb = sigsetjmp(cint.CompilerBotch, 0)) == 3) { CACHE_REGS restore_machine_regs(); /* grow stack */ recover_from_failed_susp_on_cls(&cint, 0); Yap_gcl(LOCAL_Error_Size, ap->ArityOfPE + ExtraArgs, ENV, nextop); } else if (cb == 2) { restore_machine_regs(); LOCAL_Error_Size = recover_from_failed_susp_on_cls(&cint, LOCAL_Error_Size); if (!Yap_growheap(FALSE, LOCAL_Error_Size, NULL)) { save_machine_regs(); if (ap->PredFlags & LogUpdatePredFlag) { Yap_kill_iblock((ClauseUnion *)ClauseCodeToLogUpdIndex( ap->cs.p_code.TrueCodeOfPred), NULL, ap); } else { StaticIndex *cl; cl = ClauseCodeToStaticIndex(ap->cs.p_code.TrueCodeOfPred); Yap_kill_iblock((ClauseUnion *)cl, NULL, ap); } #if defined(YAPOR) || defined(THREADS) if (ap->PredFlags & LogUpdatePredFlag && !(ap->PredFlags & ThreadLocalPredFlag) && ap->ModuleOfPred != IDB_MODULE) { ap->OpcodeOfPred = LOCKPRED_OPCODE; ap->cs.p_code.TrueCodeOfPred = ap->CodeOfPred = (yamop *)(&(ap->OpcodeOfPred)); } else { #endif ap->OpcodeOfPred = INDEX_OPCODE; ap->CodeOfPred = ap->cs.p_code.TrueCodeOfPred = (yamop *)(&(ap->OpcodeOfPred)); #if defined(YAPOR) || defined(THREADS) } #endif Yap_Error(RESOURCE_ERROR_HEAP, TermNil, LOCAL_ErrorMessage); CleanCls(&cint); return FAILCODE; } } else if (cb == 4) { restore_machine_regs(); Yap_ReleaseCMem(&cint); if (!Yap_growtrail(LOCAL_Error_Size, FALSE)) { save_machine_regs(); if (ap->PredFlags & LogUpdatePredFlag) { Yap_kill_iblock((ClauseUnion *)ClauseCodeToLogUpdIndex( ap->cs.p_code.TrueCodeOfPred), NULL, ap); } else { StaticIndex *cl; cl = ClauseCodeToStaticIndex(ap->cs.p_code.TrueCodeOfPred); Yap_kill_iblock((ClauseUnion *)cl, NULL, ap); } CleanCls(&cint); return FAILCODE; } } restart_index: cint.CodeStart = cint.cpc = cint.BlobsStart = cint.icpc = NIL; cint.CurrentPred = ap; LOCAL_ErrorMessage = NULL; LOCAL_Error_Size = 0; if (P->opc == Yap_opcode(_expand_clauses)) { expand_clauses = P; } else { expand_clauses = NULL; } #if DEBUG if (GLOBAL_Option['i' - 'a' + 1]) { Yap_DebugWriteIndicator(ap); } #endif if ((labp = expand_index(&cint)) == NULL) { if (expand_clauses) { P = FAILCODE; recover_ecls_block(expand_clauses); } Yap_ReleaseCMem(&cint); CleanCls(&cint); return FAILCODE; } if (*labp == FAILCODE) { if (expand_clauses) { P = FAILCODE; recover_ecls_block(expand_clauses); } Yap_ReleaseCMem(&cint); CleanCls(&cint); return FAILCODE; } #if DEBUG if (GLOBAL_Option['i' - 'a' + 1]) { Yap_ShowCode(&cint); } #endif /* globals for assembler */ LOCAL_IPredArity = ap->ArityOfPE; if (cint.CodeStart) { if ((indx_out = Yap_assemble(ASSEMBLING_EINDEX, TermNil, ap, FALSE, &cint, cint.i_labelno + 1)) == NULL) { if (!Yap_growheap(FALSE, LOCAL_Error_Size, NULL)) { Yap_Error(RESOURCE_ERROR_HEAP, TermNil, LOCAL_ErrorMessage); Yap_ReleaseCMem(&cint); CleanCls(&cint); return FAILCODE; } goto restart_index; } } else { /* single case */ if (expand_clauses) { P = *labp; recover_ecls_block(expand_clauses); } Yap_ReleaseCMem(&cint); CleanCls(&cint); return *labp; } if (indx_out == NULL) { if (expand_clauses) { P = FAILCODE; recover_ecls_block(expand_clauses); } Yap_ReleaseCMem(&cint); CleanCls(&cint); return FAILCODE; } Yap_ReleaseCMem(&cint); CleanCls(&cint); *labp = indx_out; if (ap->PredFlags & LogUpdatePredFlag) { /* add to head of current code children */ LogUpdIndex *ic = cint.current_cl.lui, *nic = ClauseCodeToLogUpdIndex(indx_out); if (ic == NULL) ic = (LogUpdIndex *)Yap_find_owner_index((yamop *)labp, ap); /* insert myself in the indexing code chain */ nic->SiblingIndex = ic->ChildIndex; nic->PrevSiblingIndex = NULL; if (ic->ChildIndex) { ic->ChildIndex->PrevSiblingIndex = nic; } nic->ParentIndex = ic; nic->ClFlags &= ~SwitchRootMask; ic->ChildIndex = nic; ic->ClRefCount++; } else { /* add to head of current code children */ StaticIndex *ic = cint.current_cl.si, *nic = ClauseCodeToStaticIndex(indx_out); if (ic == NULL) ic = (StaticIndex *)Yap_find_owner_index((yamop *)labp, ap); /* insert myself in the indexing code chain */ nic->SiblingIndex = ic->ChildIndex; ic->ChildIndex = nic; } if (expand_clauses) { P = indx_out; recover_ecls_block(expand_clauses); } return indx_out; } yamop *Yap_ExpandIndex(PredEntry *ap, UInt nargs) { CACHE_REGS return ExpandIndex(ap, nargs, CP PASS_REGS); } static path_stack_entry *push_path(path_stack_entry *sp, yamop **pipc, ClauseDef *clp, struct intermediates *cint) { CACHE_REGS if (sp + 1 > (path_stack_entry *)LOCAL_TrailTop) { save_machine_regs(); siglongjmp(cint->CompilerBotch, 4); } sp->flag = pc_entry; sp->uip.pce.pi_pc = pipc; sp->uip.pce.code = clp->Code; sp->uip.pce.current_code = clp->CurrentCode; sp->uip.pce.work_pc = clp->ucd.WorkPC; sp->uip.pce.tag = clp->Tag; return sp + 1; } static path_stack_entry *fetch_new_block(path_stack_entry *sp, yamop **pipc, PredEntry *ap, struct intermediates *cint) { CACHE_REGS if (sp + 1 > (path_stack_entry *)LOCAL_TrailTop) { save_machine_regs(); siglongjmp(cint->CompilerBotch, 4); } /* add current position */ sp->flag = block_entry; sp->uip.cle.entry_code = pipc; if (ap->PredFlags & LogUpdatePredFlag) { sp->uip.cle.block = (ClauseUnion *)ClauseCodeToLogUpdIndex(*pipc); } else { sp->uip.cle.block = (ClauseUnion *)ClauseCodeToStaticIndex(*pipc); } return sp + 1; } static path_stack_entry *init_block_stack(path_stack_entry *sp, yamop *ipc, PredEntry *ap) { /* add current position */ sp->flag = block_entry; sp->uip.cle.entry_code = NULL; if (ap->PredFlags & LogUpdatePredFlag) { sp->uip.cle.block = (ClauseUnion *)ClauseCodeToLogUpdIndex(ipc); } else { sp->uip.cle.block = (ClauseUnion *)ClauseCodeToStaticIndex(ipc); } return sp + 1; } static path_stack_entry *cross_block(path_stack_entry *sp, yamop **pipc, PredEntry *ap, struct intermediates *cint) { yamop *ipc = *pipc; path_stack_entry *tsp = sp; ClauseUnion *block; do { UInt bsize; while ((--tsp)->flag != block_entry) ; block = tsp->uip.cle.block; if (block->lui.ClFlags & LogUpdMask) bsize = block->lui.ClSize; else bsize = block->si.ClSize; if (ipc > (yamop *)block && ipc < (yamop *)((CODEADDR)block + bsize)) { path_stack_entry *nsp = tsp + 1; for (; tsp < sp; tsp++) { if (tsp->flag == pc_entry) { if (nsp != tsp) { nsp->flag = pc_entry; nsp->uip.pce.pi_pc = tsp->uip.pce.pi_pc; nsp->uip.pce.code = tsp->uip.pce.code; nsp->uip.pce.current_code = tsp->uip.pce.current_code; nsp->uip.pce.work_pc = tsp->uip.pce.work_pc; nsp->uip.pce.tag = tsp->uip.pce.tag; } nsp++; } } return nsp; } } while (tsp->uip.cle.entry_code != NULL); /* moved to a new block */ return fetch_new_block(sp, pipc, ap, cint); } static yamop *pop_path(path_stack_entry **spp, ClauseDef *clp, PredEntry *ap, struct intermediates *cint) { path_stack_entry *sp = *spp; yamop *nipc; while ((--sp)->flag != pc_entry) ; *spp = sp; clp->Code = sp->uip.pce.code; clp->CurrentCode = sp->uip.pce.current_code; clp->ucd.WorkPC = sp->uip.pce.work_pc; clp->Tag = sp->uip.pce.tag; if (sp->uip.pce.pi_pc == NULL) { *spp = sp; return NULL; } nipc = *(sp->uip.pce.pi_pc); *spp = cross_block(sp, sp->uip.pce.pi_pc, ap, cint); return nipc; } static int table_fe_overflow(yamop *pc, Functor f) { if (pc->y_u.sssl.s <= MIN_HASH_ENTRIES) { /* we cannot expand otherwise */ COUNT i; FuncSwiEntry *csw = (FuncSwiEntry *)pc->y_u.sssl.l; for (i = 0; i < pc->y_u.sssl.s; i++, csw++) { if (csw->Tag == f) return FALSE; } return TRUE; } else { COUNT free = pc->y_u.sssl.s - pc->y_u.sssl.e; return (!free || pc->y_u.sssl.s / free > 4); } } static int table_ae_overflow(yamop *pc, Term at) { if (pc->y_u.sssl.s <= MIN_HASH_ENTRIES) { /* check if we are already there */ COUNT i; AtomSwiEntry *csw = (AtomSwiEntry *)pc->y_u.sssl.l; for (i = 0; i < pc->y_u.sssl.s; i++, csw++) { if (csw->Tag == at) return FALSE; } return TRUE; } else { COUNT free = pc->y_u.sssl.s - pc->y_u.sssl.e; return (!free || pc->y_u.sssl.s / free > 4); } } static void replace_index_block(ClauseUnion *parent_block, yamop *cod, yamop *ncod, PredEntry *ap) { if (ap->PredFlags & LogUpdatePredFlag) { LogUpdIndex *cl = ClauseCodeToLogUpdIndex(cod), *ncl = ClauseCodeToLogUpdIndex(ncod), *c = parent_block->lui.ChildIndex; ncl->SiblingIndex = cl->SiblingIndex; ncl->PrevSiblingIndex = cl->PrevSiblingIndex; ncl->ClRefCount = cl->ClRefCount; ncl->ChildIndex = cl->ChildIndex; ncl->ParentIndex = cl->ParentIndex; ncl->ClPred = cl->ClPred; // INIT_LOCK(ncl->ClLock); if (c == cl) { parent_block->lui.ChildIndex = ncl; } else { if (cl->PrevSiblingIndex) cl->PrevSiblingIndex->SiblingIndex = ncl; } if (cl->SiblingIndex) { cl->SiblingIndex->PrevSiblingIndex = ncl; } c = cl->ChildIndex; while (c != NULL) { c->ParentIndex = ncl; c = c->SiblingIndex; } Yap_InformOfRemoval(cl); Yap_LUIndexSpace_SW -= cl->ClSize; Yap_FreeCodeSpace((char *)cl); } else { StaticIndex *cl = ClauseCodeToStaticIndex(cod), *ncl = ClauseCodeToStaticIndex(ncod), *c = parent_block->si.ChildIndex; ncl->SiblingIndex = cl->SiblingIndex; ncl->ClPred = cl->ClPred; if (c == cl) { parent_block->si.ChildIndex = ncl; } else { while (c->SiblingIndex != cl) { c = c->SiblingIndex; } c->SiblingIndex = ncl; } Yap_InformOfRemoval(cl); Yap_IndexSpace_SW -= cl->ClSize; Yap_FreeCodeSpace((char *)cl); } } static AtomSwiEntry *expand_ctable(yamop *pc, ClauseUnion *blk, struct intermediates *cint, Term at) { PredEntry *ap = cint->CurrentPred; int n = pc->y_u.sssl.s, i, i0 = n; UInt fail_l = Zero; AtomSwiEntry *old_ae = (AtomSwiEntry *)(pc->y_u.sssl.l), *target; if (n > MIN_HASH_ENTRIES) { AtomSwiEntry *tmp = old_ae; int i; n = 1; for (i = 0; i < pc->y_u.sssl.s; i++, tmp++) { if (tmp->Tag != Zero) n++; else fail_l = tmp->u_a.Label; } } else { fail_l = old_ae[n].u_a.Label; n++; } if (n > MIN_HASH_ENTRIES) { int cases = MIN_HASH_ENTRIES, i, n0; n0 = n + 1 + n / 4; while (cases < n0) cases *= 2; if (cases == pc->y_u.sssl.s) { return fetch_centry(old_ae, at, n - 1, n); } /* initialize */ target = (AtomSwiEntry *)emit_switch_space(cases, sizeof(AtomSwiEntry), cint, 0); pc->opc = Yap_opcode(_switch_on_cons); pc->y_u.sssl.s = cases; for (i = 0; i < cases; i++) { target[i].Tag = Zero; target[i].u_a.Label = fail_l; } } else { pc->opc = Yap_opcode(_if_cons); pc->y_u.sssl.s = n; target = (AtomSwiEntry *)emit_switch_space(n + 1, sizeof(AtomSwiEntry), cint, 0); target[n].Tag = Zero; target[n].u_a.Label = fail_l; } for (i = 0; i < i0; i++, old_ae++) { Term tag = old_ae->Tag; if (tag != Zero) { AtomSwiEntry *ics = fetch_centry(target, tag, i, n); ics->Tag = tag; ics->u_a.Label = old_ae->u_a.Label; } } /* support for threads */ if (blk) replace_index_block(blk, pc->y_u.sssl.l, (yamop *)target, ap); pc->y_u.sssl.l = (yamop *)target; return fetch_centry(target, at, n - 1, n); } static FuncSwiEntry *expand_ftable(yamop *pc, ClauseUnion *blk, struct intermediates *cint, Functor f) { PredEntry *ap = cint->CurrentPred; int n = pc->y_u.sssl.s, i, i0 = n; UInt fail_l = Zero; FuncSwiEntry *old_fe = (FuncSwiEntry *)(pc->y_u.sssl.l), *target; if (n > MIN_HASH_ENTRIES) { FuncSwiEntry *tmp = old_fe; int i; n = 1; for (i = 0; i < pc->y_u.sssl.s; i++, tmp++) { if (tmp->Tag != Zero) n++; else fail_l = tmp->u_f.Label; } } else { fail_l = old_fe[n].u_f.Label; n++; } if (n > MIN_HASH_ENTRIES) { int cases = MIN_HASH_ENTRIES, i, n0; n0 = n + 1 + n / 4; while (cases < n0) cases *= 2; if (cases == pc->y_u.sssl.s) { return fetch_fentry(old_fe, f, n - 1, n); } pc->opc = Yap_opcode(_switch_on_func); pc->y_u.sssl.s = cases; pc->y_u.sssl.e = n; pc->y_u.sssl.w = 0; /* initialize */ target = (FuncSwiEntry *)emit_switch_space(cases, sizeof(FuncSwiEntry), cint, FuncSwitchMask); for (i = 0; i < cases; i++) { target[i].Tag = NULL; target[i].u_f.Label = fail_l; } } else { pc->opc = Yap_opcode(_if_func); pc->y_u.sssl.s = n; pc->y_u.sssl.e = n; pc->y_u.sssl.w = 0; target = (FuncSwiEntry *)emit_switch_space(n + 1, sizeof(FuncSwiEntry), cint, FuncSwitchMask); target[n].Tag = Zero; target[n].u_f.Label = fail_l; } for (i = 0; i < i0; i++, old_fe++) { Functor f = old_fe->Tag; if (f != NULL) { FuncSwiEntry *ifs = fetch_fentry(target, f, i, n); ifs->Tag = old_fe->Tag; ifs->u_f.Label = old_fe->u_f.Label; } } replace_index_block(blk, pc->y_u.sssl.l, (yamop *)target, ap); pc->y_u.sssl.l = (yamop *)target; return fetch_fentry(target, f, n - 1, n); } static void clean_ref_to_clause(LogUpdClause *tgl) { tgl->ClRefCount--; if ((tgl->ClFlags & ErasedMask) && !(tgl->ClRefCount) && !(tgl->ClFlags & InUseMask)) { /* last ref to the clause */ Yap_ErLogUpdCl(tgl); } } static ClauseUnion *current_block(path_stack_entry *sp) { while ((--sp)->flag != block_entry) ; return sp->uip.cle.block; } static path_stack_entry *kill_block(path_stack_entry *sp, PredEntry *ap) { while ((--sp)->flag != block_entry) ; if (sp->uip.cle.entry_code == NULL) { Yap_kill_iblock(sp->uip.cle.block, NULL, ap); } else { path_stack_entry *nsp = sp; while ((--nsp)->flag != block_entry) ; Yap_kill_iblock(sp->uip.cle.block, nsp->uip.cle.block, ap); *sp->uip.cle.entry_code = (yamop *)&(ap->cs.p_code.ExpandCode); } return sp; } static LogUpdClause *find_last_clause(yamop *start) { while (start->y_u.OtaLl.d->ClFlags & ErasedMask) start = start->y_u.OtaLl.n; /* this should be the available clause */ return start->y_u.OtaLl.d; } static void remove_clause_from_index(yamop *header, LogUpdClause *cl) { yamop **prevp = &(header->y_u.Illss.l1); yamop *curp = header->y_u.Illss.l1; if (curp->y_u.OtaLl.d == cl) { yamop *newp = curp->y_u.OtaLl.n; newp->opc = curp->opc; *prevp = newp; } else { yamop *ocurp = NULL, *ocurp0 = curp; while (curp->y_u.OtaLl.d != cl) { ocurp = curp; curp = curp->y_u.OtaLl.n; } /* in case we were the last */ if (curp == header->y_u.Illss.l2) header->y_u.Illss.l2 = ocurp; if (ocurp != ocurp0) ocurp->opc = curp->opc; ocurp->y_u.OtILl.n = curp->y_u.OtaLl.n; ocurp->y_u.OtILl.block = curp->y_u.OtILl.block; } header->y_u.Illss.e--; #if DEBUG Yap_DirtyCps--; Yap_FreedCps++; #endif clean_ref_to_clause(cl); Yap_LUIndexSpace_CP -= (UInt)NEXTOP((yamop *)NULL, OtILl); Yap_FreeCodeSpace((ADDR)curp); } static void remove_dirty_clauses_from_index(yamop *header) { LogUpdClause *cl; yamop *previouscurp; OPCODE endop = Yap_opcode(_trust_logical); yamop **prevp = &(header->y_u.Illss.l1), *curp = header->y_u.Illss.l1; OPCODE startopc = curp->opc; PredEntry *ap = curp->y_u.OtaLl.d->ClPred; if (ap->PredFlags & CountPredFlag) endop = Yap_opcode(_count_trust_logical); else if (ap->PredFlags & ProfiledPredFlag) endop = Yap_opcode(_profiled_trust_logical); while ((cl = curp->y_u.OtaLl.d) && (cl->ClFlags & ErasedMask)) { yamop *ocurp = curp; header->y_u.Illss.e--; #if DEBUG Yap_DirtyCps--; Yap_FreedCps++; #endif // if (ap->ModuleOfPred!=IDB_MODULE && // !strcmp(RepAtom(NameOfFunctor(ap->FunctorOfPred))->StrOfAE, // "$lgt_send_to_obj_ne_")) // printf(" L %p %p %d %p\n", curp, curp->y_u.OtaLl.n, header->y_u.Illss.e, // curp->opc); clean_ref_to_clause(cl); curp = curp->y_u.OtaLl.n; Yap_LUIndexSpace_CP -= (UInt)NEXTOP((yamop *)NULL, OtaLl); Yap_FreeCodeSpace((ADDR)ocurp); if (ocurp == header->y_u.Illss.l2) { LogUpdIndex *clau = header->y_u.Illss.I; /* no clauses left */ if (clau->ClFlags & ErasedMask) { Yap_ErLogUpdIndex(clau); return; } header->y_u.Illss.l1 = header->y_u.Illss.l2 = NULL; header->y_u.Illss.s = header->y_u.Illss.e = 0; return; } } *prevp = curp; curp->opc = startopc; if (curp->opc == endop) return; // don't try to follow the chain if there is no chain. if (header->y_u.Illss.e <= 1) return; previouscurp = curp; curp = curp->y_u.OtaLl.n; while (TRUE) { if ((cl = curp->y_u.OtaLl.d)->ClFlags & ErasedMask) { yamop *ocurp = curp; header->y_u.Illss.e--; #if DEBUG Yap_DirtyCps--; Yap_FreedCps++; #endif clean_ref_to_clause(cl); if (curp->opc == endop) { previouscurp->opc = endop; previouscurp->y_u.OtILl.block = curp->y_u.OtILl.block; previouscurp->y_u.OtILl.n = NULL; header->y_u.Illss.l2 = previouscurp; Yap_LUIndexSpace_CP -= (UInt)NEXTOP((yamop *)NULL, OtILl); Yap_FreeCodeSpace((ADDR)curp); return; } previouscurp->y_u.OtaLl.n = curp->y_u.OtaLl.n; curp = curp->y_u.OtaLl.n; Yap_LUIndexSpace_CP -= (UInt)NEXTOP((yamop *)NULL, OtaLl); Yap_FreeCodeSpace((ADDR)ocurp); if (!header->y_u.Illss.e) return; } else { previouscurp = curp; if (curp->opc == endop) { curp->y_u.OtILl.n = NULL; return; } curp = curp->y_u.OtaLl.n; } } } static path_stack_entry *kill_clause(yamop *ipc, yamop *bg, yamop *lt, path_stack_entry *sp0, PredEntry *ap) { LogUpdIndex *blk; yamop *start; op_numbers op0; path_stack_entry *sp = sp0; while ((--sp)->flag != block_entry) ; blk = (LogUpdIndex *)(sp->uip.cle.block); start = blk->ClCode; op0 = Yap_op_from_opcode(start->opc); while (op0 == _lock_lu) { start = NEXTOP(start, p); op0 = Yap_op_from_opcode(start->opc); } while (op0 == _jump_if_nonvar) { start = NEXTOP(start, xll); op0 = Yap_op_from_opcode(start->opc); } if (op0 != _enter_lu_pred) { /* static code */ return kill_block(sp + 1, ap); } /* weird case ????? */ if (!start->y_u.Illss.s) { /* ERROR */ Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "Illss.s == 0 %p", ipc); return sp; } if (start->y_u.Illss.s == 1) { /* we need to discover which clause is left and then die */ path_stack_entry *nsp; find_last_clause(start->y_u.Illss.l1); nsp = sp; while ((--nsp)->flag != block_entry) ; /* make us point straight at clause */ *sp->uip.cle.entry_code = FAILCODE; Yap_kill_iblock(sp->uip.cle.block, nsp->uip.cle.block, ap); return sp; } else { if ( #if MULTIPLE_STACKS blk->ClRefCount == 0 #else !(blk->ClFlags & InUseMask) #endif ) { remove_clause_from_index(start, ClauseCodeToLogUpdClause(bg)); } else { blk->ClFlags |= DirtyMask; } return sp; } } static path_stack_entry *expanda_block(path_stack_entry *sp, PredEntry *ap, ClauseDef *cls, int group1, yamop *alt, struct intermediates *cint) { while ((--sp)->flag != block_entry) ; Yap_kill_iblock(sp->uip.cle.block, NULL, ap); return sp; } static path_stack_entry *expandz_block(path_stack_entry *sp, PredEntry *ap, ClauseDef *cls, int group1, yamop *alt, struct intermediates *cint) { while ((--sp)->flag != block_entry) ; Yap_kill_iblock(sp->uip.cle.block, NULL, ap); return sp; } static LogUpdClause *lu_clause(yamop *ipc, PredEntry *ap) { if (ipc == FAILCODE) return NULL; if (ipc == (yamop *)(&(ap->OpcodeOfPred))) return NULL; return ClauseCodeToLogUpdClause(ipc); } static StaticClause *find_static_clause(PredEntry *ap, yamop *ipc) { StaticClause *cl = ClauseCodeToStaticClause(ap->cs.p_code.FirstClause); while (ipc < cl->ClCode || ipc > (yamop *)((char *)cl + cl->ClSize)) { cl = cl->ClNext; if (!cl) return NULL; } return cl; } static StaticClause *static_clause(yamop *ipc, PredEntry *ap, int trust) { CELL *p; if (ipc == FAILCODE) return NULL; if (ipc == (yamop *)(&(ap->OpcodeOfPred))) return NULL; if (ap->PredFlags & MegaClausePredFlag) return (StaticClause *)ipc; if (ap->PredFlags & TabledPredFlag) ipc = PREVOP(ipc, Otapl); p = (CELL *)ipc; if (trust) { return ClauseCodeToStaticClause(p); } else { op_numbers op = Yap_op_from_opcode(ipc->opc); UInt j; /* unbound call, so we cannot optimise instructions */ switch (op) { case _p_db_ref_x: case _p_float_x: j = Yap_regnotoreg(ipc->y_u.xl.x); break; case _get_list: j = Yap_regnotoreg(ipc->y_u.x.x); break; case _get_atom: j = Yap_regnotoreg(ipc->y_u.xc.x); break; case _get_float: j = Yap_regnotoreg(ipc->y_u.xd.x); break; case _get_struct: j = Yap_regnotoreg(ipc->y_u.xd.x); break; case _get_2atoms: case _get_3atoms: case _get_4atoms: case _get_5atoms: case _get_6atoms: return ClauseCodeToStaticClause(p); default: return find_static_clause(ap, ipc); } if (j == 1) /* must be the first instruction */ return ClauseCodeToStaticClause(p); return find_static_clause(ap, ipc); } return NULL; } static StaticClause *simple_static_clause(yamop *ipc, PredEntry *ap) { if (ipc == (yamop *)(&(ap->OpcodeOfPred))) return NULL; if (ipc == FAILCODE) return NULL; return ClauseCodeToStaticClause(ipc); } /* this code should be called when we jumped to clauses */ static path_stack_entry *kill_unsafe_block(path_stack_entry *sp, op_numbers op, PredEntry *ap, int first, int remove, ClauseDef *cls) { yamop *ipc; while ((--sp)->flag != block_entry) ; if (sp->uip.cle.entry_code == NULL) { /* we have reached the top */ Yap_RemoveIndexation(ap); return sp; } ipc = *sp->uip.cle.entry_code; if (Yap_op_from_opcode(ipc->opc) == op) { /* the new block was the current clause */ ClauseDef cld[2]; if (remove) { *sp->uip.cle.entry_code = FAILCODE; return sp; } if (ap->PredFlags & LogUpdatePredFlag) { struct intermediates intrs; LogUpdClause *lc = lu_clause(ipc, ap); if (first) { cld[0].Code = cls[0].Code; cld[1].Code = lc->ClCode; } else { cld[0].Code = lc->ClCode; cld[1].Code = cls[0].Code; } intrs.expand_block = NULL; *sp->uip.cle.entry_code = (yamop *)suspend_indexing(cld, cld + 1, ap, &intrs); } else { /* static predicate, shouldn't do much, just suspend the code here */ *sp->uip.cle.entry_code = (yamop *)&(ap->cs.p_code.ExpandCode); return sp; } return sp; } /* we didn't have protection, should kill now */ return kill_block(sp + 1, ap); } static int compacta_expand_clauses(yamop *ipc) { /* expand clauses so that you have a hole at the beginning */ /* we know that there is at least one element here */ yamop **start = (yamop **)(NEXTOP(ipc, sssllp)); yamop **ptr, **end; ptr = end = start + ipc->y_u.sssllp.s1; while (ptr > start) { yamop *next = *--ptr; if (next) *--end = next; } if (ptr != end) { while (end > start) { *--end = NULL; } return TRUE; } return FALSE; } static int compactz_expand_clauses(yamop *ipc) { /* expand clauses so that you have a hole at the beginning */ /* we know that there is at least one element here */ yamop **start = (yamop **)(NEXTOP(ipc, sssllp)); yamop **ptr, **end; end = start + ipc->y_u.sssllp.s1; ptr = start; while (ptr < end) { yamop *next = *ptr++; if (next) *start++ = next; } /* reset empty slots at end */ if (start != end) { while (start < end) { *start++ = NULL; } return TRUE; } return FALSE; } /* this code should be called when we jumped to clauses */ static yamop *add_to_expand_clauses(path_stack_entry **spp, yamop *ipc, ClauseDef *cls, PredEntry *ap, int first, struct intermediates *cint) { path_stack_entry *sp = *spp; yamop **clar; if (first) { do { clar = (yamop **)NEXTOP(ipc, sssllp); if (*clar == NULL || clar[0] == cls->Code) { while (*clar == NULL) clar++; if (clar[0] != cls->Code) { clar[-1] = cls->Code; ipc->y_u.sssllp.s2++; } return pop_path(spp, cls, ap, cint); } } while (compacta_expand_clauses(ipc)); } else { do { clar = (yamop **)NEXTOP(ipc, sssllp) + ipc->y_u.sssllp.s1; if (clar[-1] == NULL || clar[-1] == cls->Code) { while (*--clar == NULL) ; if (clar[0] != cls->Code) { clar[1] = cls->Code; ipc->y_u.sssllp.s2++; } return pop_path(spp, cls, ap, cint); } } while (compactz_expand_clauses(ipc)); } while ((--sp)->flag != block_entry) ; if (sp->uip.cle.entry_code) { *sp->uip.cle.entry_code = (yamop *)&(ap->cs.p_code.ExpandCode); } recover_ecls_block(ipc); return pop_path(spp, cls, ap, cint); } /* this code should be called when we jumped to clauses */ static void nullify_expand_clause(yamop *ipc, path_stack_entry *sp, ClauseDef *cls) { yamop **st = (yamop **)NEXTOP(ipc, sssllp); yamop **max = st + ipc->y_u.sssllp.s1; /* make sure we get rid of the reference */ while (st < max) { if (*st && *st == cls->Code) { *st = NULL; ipc->y_u.sssllp.s2--; break; } st++; } /* if the block has a single element */ if (ipc->y_u.sssllp.s2 == 1) { yamop **st = (yamop **)NEXTOP(ipc, sssllp); while ((--sp)->flag != block_entry) ; while (TRUE) { if (*st && *st != cls->Code) { *sp->uip.cle.entry_code = *st; recover_ecls_block(ipc); return; } st++; } } } static yamop *add_try(PredEntry *ap, ClauseDef *cls, yamop *next, struct intermediates *cint) { yamop *newcp; UInt size = (UInt)NEXTOP((yamop *)NULL, OtaLl); LogUpdClause *lcl = ClauseCodeToLogUpdClause(cls->Code); if ((newcp = (yamop *)Yap_AllocCodeSpace(size)) == NULL) { /* OOOPS, got in trouble, must do a siglongjmp and recover space */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 2); } Yap_LUIndexSpace_CP += size; #if DEBUG Yap_NewCps++; Yap_LiveCps++; #endif newcp->opc = Yap_opcode(_try_logical); newcp->y_u.OtaLl.s = ap->ArityOfPE; newcp->y_u.OtaLl.n = next; newcp->y_u.OtaLl.d = lcl; lcl->ClRefCount++; return newcp; } static yamop *add_trust(LogUpdIndex *icl, ClauseDef *cls, struct intermediates *cint) { yamop *newcp; UInt size = (UInt)NEXTOP((yamop *)NULL, OtILl); LogUpdClause *lcl = ClauseCodeToLogUpdClause(cls->Code); PredEntry *ap = lcl->ClPred; if ((newcp = (yamop *)Yap_AllocCodeSpace(size)) == NULL) { /* OOOPS, got in trouble, must do a siglongjmp and recover space */ save_machine_regs(); siglongjmp(cint->CompilerBotch, 2); } Yap_LUIndexSpace_CP += size; #if DEBUG Yap_NewCps++; Yap_LiveCps++; #endif if (ap->PredFlags & CountPredFlag) newcp->opc = Yap_opcode(_count_trust_logical); else if (ap->PredFlags & ProfiledPredFlag) newcp->opc = Yap_opcode(_profiled_trust_logical); else newcp->opc = Yap_opcode(_trust_logical); newcp->y_u.OtILl.block = icl; newcp->y_u.OtILl.n = NULL; newcp->y_u.OtILl.d = lcl; lcl->ClRefCount++; return newcp; } static void add_to_index(struct intermediates *cint, int first, path_stack_entry *sp, ClauseDef *cls) { /* last clause to experiment with */ PredEntry *ap = cint->CurrentPred; yamop *ipc = ap->cs.p_code.TrueCodeOfPred; int group1 = TRUE; yamop *alt = NULL; UInt current_arity = 0; LogUpdIndex *icl = NULL; sp = init_block_stack(sp, ipc, ap); /* try to refine the interval using the indexing code */ while (ipc != NULL) { op_numbers op = Yap_op_from_opcode(ipc->opc); switch (op) { case _try_logical: case _retry_logical: case _count_retry_logical: case _profiled_retry_logical: case _trust_logical: case _count_trust_logical: case _profiled_trust_logical: /* ERROR */ break; case _enter_lu_pred: ipc->y_u.Illss.s++; icl = ipc->y_u.Illss.I; if (first) { if (ap->PredFlags & CountPredFlag) ipc->y_u.Illss.l1->opc = Yap_opcode(_count_retry_logical); else if (ap->PredFlags & ProfiledPredFlag) ipc->y_u.Illss.l1->opc = Yap_opcode(_profiled_retry_logical); else ipc->y_u.Illss.l1->opc = Yap_opcode(_retry_logical); ipc->y_u.Illss.l1 = add_try(ap, cls, ipc->y_u.Illss.l1, cint); } else { /* just go to next instruction */ yamop *end = add_trust(icl, cls, cint), *old = ipc->y_u.Illss.l2; /* we used to have two clauses */ if (ap->PredFlags & CountPredFlag) old->opc = Yap_opcode(_count_retry_logical); else if (ap->PredFlags & ProfiledPredFlag) old->opc = Yap_opcode(_profiled_retry_logical); else old->opc = Yap_opcode(_retry_logical); old->y_u.OtaLl.n = end; old->y_u.OtaLl.s = ap->ArityOfPE; ipc->y_u.Illss.l2 = end; } ipc = pop_path(&sp, cls, ap, cint); break; case _try_clause: /* I cannot expand a predicate that starts on a variable, have to expand the index. */ if (first) { sp = expanda_block(sp, ap, cls, group1, alt, cint); ipc = pop_path(&sp, cls, ap, cint); } else { /* just go to next instruction */ ipc = NEXTOP(ipc, Otapl); } break; case _try_clause2: case _try_clause3: case _try_clause4: /* I cannot expand a predicate that starts on a variable, have to expand the index. */ if (first) { sp = expanda_block(sp, ap, cls, group1, alt, cint); ipc = pop_path(&sp, cls, ap, cint); } else { /* just go to next instruction */ ipc = NEXTOP(ipc, l); } break; case _retry: /* this clause had no indexing */ ipc = NEXTOP(ipc, Otapl); break; case _retry2: case _retry3: case _retry4: /* this clause had no indexing */ ipc = NEXTOP(ipc, l); break; /* instructions type l */ case _retry_me: /* should never be reached both for asserta */ group1 = FALSE; ipc = ipc->y_u.Otapl.d; break; case _try_me: if (first) { ipc = NEXTOP(ipc, Otapl); alt = ipc->y_u.Otapl.d; } else { ipc = ipc->y_u.Otapl.d; group1 = FALSE; } break; case _retry_profiled: case _count_retry: ipc = NEXTOP(ipc, Otapl); break; case _profiled_trust_me: case _trust_me: case _count_trust_me: group1 = FALSE; ipc = NEXTOP(ipc, Otapl); break; case _trust: sp = expandz_block(sp, ap, cls, group1, alt, cint); ipc = pop_path(&sp, cls, ap, cint); break; case _jump: sp = cross_block(sp, &ipc->y_u.l.l, ap, cint); /* just skip for now, but should worry about memory management */ ipc = ipc->y_u.l.l; break; case _jump_if_var: sp = push_path(sp, &(ipc->y_u.l.l), cls, cint); ipc = NEXTOP(ipc, l); break; case _jump_if_nonvar: sp = push_path(sp, &(ipc->y_u.xll.l2), cls, cint); sp = cross_block(sp, &ipc->y_u.xll.l1, ap, cint); ipc = ipc->y_u.xll.l1; break; /* instructions type EC */ case _try_in: /* we are done */ if (first) { sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { ipc = NEXTOP(ipc, l); } break; case _user_switch: ipc = ipc->y_u.lp.l; break; /* instructions type e */ case _switch_on_type: sp = push_path(sp, &(ipc->y_u.llll.l4), cls, cint); if (ap->PredFlags & LogUpdatePredFlag) { add_head_info(cls, 1); } else { add_info(cls, 1); } if (IsPairTerm(cls->Tag)) { yamop *nipc = ipc->y_u.llll.l1; current_arity = 2; move_next(cls, 1); if (nipc == FAILCODE) { /* jump straight to clause */ if (ap->PredFlags & LogUpdatePredFlag) { ipc->y_u.llll.l1 = cls->Code; } else { ipc->y_u.llll.l1 = cls->CurrentCode; } ipc = pop_path(&sp, cls, ap, cint); } else { /* go on */ sp = cross_block(sp, &ipc->y_u.llll.l1, ap, cint); ipc = nipc; } } else if (IsAtomOrIntTerm(cls->Tag)) { yamop *nipc = ipc->y_u.llll.l2; move_next(cls, 1); if (nipc == FAILCODE) { /* need to expand the block */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else if (IsApplTerm(cls->Tag)) { yamop *nipc = ipc->y_u.llll.l3; if (nipc == FAILCODE) { /* need to expand the block */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else { /* we can't separate into four groups, need to restart. */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } break; case _switch_list_nl: sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); break; case _switch_on_arg_type: sp = push_path(sp, &(ipc->y_u.xllll.l4), cls, cint); if (ap->PredFlags & LogUpdatePredFlag) { add_head_info(cls, Yap_regtoregno(ipc->y_u.xllll.x)); } else { add_info(cls, Yap_regtoregno(ipc->y_u.xllll.x)); } if (IsPairTerm(cls->Tag)) { yamop *nipc = ipc->y_u.xllll.l1; current_arity = 2; move_next(cls, Yap_regtoregno(ipc->y_u.xllll.x)); if (nipc == FAILCODE) { /* jump straight to clause */ if (ap->PredFlags & LogUpdatePredFlag) { ipc->y_u.xllll.l1 = cls->Code; } else { ipc->y_u.xllll.l1 = cls->CurrentCode; } ipc = pop_path(&sp, cls, ap, cint); } else { /* go on */ sp = cross_block(sp, &ipc->y_u.xllll.l1, ap, cint); ipc = nipc; } } else if (IsAtomOrIntTerm(cls->Tag)) { yamop *nipc = ipc->y_u.xllll.l2; move_next(cls, Yap_regtoregno(ipc->y_u.xllll.x)); if (nipc == FAILCODE) { /* need to expand the block */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else if (IsApplTerm(cls->Tag)) { yamop *nipc = ipc->y_u.xllll.l3; move_next(cls, Yap_regtoregno(ipc->y_u.xllll.x)); if (nipc == FAILCODE) { /* need to expand the block */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else { /* we can't separate into four groups, need to restart. */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } break; case _switch_on_sub_arg_type: sp = push_path(sp, &(ipc->y_u.sllll.l4), cls, cint); add_arg_info(cls, ap, ipc->y_u.sllll.s + 1); if (IsPairTerm(cls->Tag)) { yamop *nipc = ipc->y_u.sllll.l1; current_arity = 2; skip_to_arg(cls, ap, ipc->y_u.sllll.s, current_arity); if (nipc == FAILCODE) { /* jump straight to clause */ if (ap->PredFlags & LogUpdatePredFlag) { ipc->y_u.sllll.l1 = cls->Code; } else { ipc->y_u.sllll.l1 = cls->CurrentCode; } ipc = pop_path(&sp, cls, ap, cint); } else { /* go on */ sp = cross_block(sp, &ipc->y_u.sllll.l1, ap, cint); ipc = nipc; } } else if (IsAtomOrIntTerm(cls->Tag)) { yamop *nipc = ipc->y_u.sllll.l2; skip_to_arg(cls, ap, ipc->y_u.sllll.s, current_arity); if (nipc == FAILCODE) { /* need to expand the block */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else if (IsApplTerm(cls->Tag)) { yamop *nipc = ipc->y_u.sllll.l3; skip_to_arg(cls, ap, ipc->y_u.sllll.s, current_arity); if (nipc == FAILCODE) { /* need to expand the block */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else { /* we can't separate into four groups, need to restart. */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } break; case _if_not_then: ipc = pop_path(&sp, cls, ap, cint); break; /* instructions type ollll */ case _switch_on_func: case _if_func: case _go_on_func: { FuncSwiEntry *fe; yamop *newpc; Functor f = (Functor)RepAppl(cls->Tag); if (op == _switch_on_func) { fe = lookup_f_hash(f, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } else { fe = lookup_f(f, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } if (!IsExtensionFunctor(f)) { current_arity = ArityOfFunctor(f); } newpc = fe->u_f.labp; if (newpc == (yamop *)&(ap->cs.p_code.ExpandCode)) { /* we found it */ ipc = pop_path(&sp, cls, ap, cint); } else if (newpc == FAILCODE) { /* oops, nothing there */ if (fe->Tag != f) { if (IsExtensionFunctor(f)) { sp = kill_unsafe_block(sp, op, ap, first, FALSE, cls); ipc = pop_path(&sp, cls, ap, cint); break; } if (table_fe_overflow(ipc, f)) { fe = expand_ftable(ipc, current_block(sp), cint, f); } fe->Tag = f; ipc->y_u.sssl.e++; } if (ap->PredFlags & LogUpdatePredFlag) { fe->u_f.labp = cls->Code; } else { fe->u_f.labp = cls->CurrentCode; } ipc = pop_path(&sp, cls, ap, cint); } else { yamop *newpc = fe->u_f.labp; sp = fetch_new_block(sp, &(ipc->y_u.sssl.l), ap, cint); sp = cross_block(sp, &(fe->u_f.labp), ap, cint); ipc = newpc; } } break; case _index_dbref: cls->Tag = cls->ucd.t_ptr; ipc = NEXTOP(ipc, e); break; case _index_blob: cls->Tag = Yap_Double_key(cls->ucd.t_ptr); ipc = NEXTOP(ipc, e); break; case _index_long: cls->Tag = Yap_Int_key(cls->ucd.t_ptr); ipc = NEXTOP(ipc, e); break; case _switch_on_cons: case _if_cons: case _go_on_cons: { AtomSwiEntry *ae; yamop *newpc; Term at = cls->Tag; if (op == _switch_on_cons) { ae = lookup_c_hash(at, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } else { ae = lookup_c(at, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } newpc = ae->u_a.labp; if (newpc == (yamop *)&(ap->cs.p_code.ExpandCode)) { /* nothing more to do */ ipc = pop_path(&sp, cls, ap, cint); } else if (newpc == FAILCODE) { /* oops, nothing there */ if (ae->Tag != at) { if (table_ae_overflow(ipc, at)) { ae = expand_ctable(ipc, current_block(sp), cint, at); } ae->Tag = at; ipc->y_u.sssl.e++; } if (ap->PredFlags & LogUpdatePredFlag) { ae->u_a.labp = cls->Code; } else { ae->u_a.labp = cls->CurrentCode; } ipc = pop_path(&sp, cls, ap, cint); } else { yamop *newpc = ae->u_a.labp; sp = fetch_new_block(sp, &(ipc->y_u.sssl.l), ap, cint); sp = cross_block(sp, &(ae->u_a.labp), ap, cint); ipc = newpc; } } break; case _expand_clauses: ipc = add_to_expand_clauses(&sp, ipc, cls, ap, first, cint); break; case _expand_index: ipc = pop_path(&sp, cls, ap, cint); break; case _lock_lu: ipc = NEXTOP(ipc, p); break; case _op_fail: while ((--sp)->flag != block_entry) ; *sp->uip.cle.entry_code = cls->Code; ipc = pop_path(&sp, cls, ap, cint); break; default: sp = kill_unsafe_block(sp, op, ap, first, FALSE, cls); ipc = pop_path(&sp, cls, ap, cint); } } } void Yap_AddClauseToIndex(PredEntry *ap, yamop *beg, int first) { CACHE_REGS ClauseDef cl; /* first clause */ path_stack_entry *stack, *sp; int cb; struct intermediates cint; if (!(ap->PredFlags & LogUpdatePredFlag)) { if (ap->PredFlags & IndexedPredFlag) Yap_RemoveIndexation(ap); return; } cint.CurrentPred = ap; cint.expand_block = NULL; cint.CodeStart = cint.BlobsStart = cint.cpc = cint.icpc = NIL; cint.term_depth = cint.last_index_new_depth = cint.last_depth_size = 0L; if ((cb = sigsetjmp(cint.CompilerBotch, 0)) == 3) { restore_machine_regs(); Yap_gcl(LOCAL_Error_Size, ap->ArityOfPE, ENV, CP); save_machine_regs(); } else if (cb == 2) { restore_machine_regs(); Yap_growheap(FALSE, LOCAL_Error_Size, NULL); save_machine_regs(); } else if (cb == 4) { restore_machine_regs(); Yap_growtrail(LOCAL_Error_Size, FALSE); save_machine_regs(); } if (cb) { Yap_RemoveIndexation(ap); return; } LOCAL_Error_Size = 0; LOCAL_ErrorMessage = NULL; #if DEBUG if (GLOBAL_Option['i' - 'a' + 1]) { Yap_DebugPutc(stderr, '+'); Yap_DebugWriteIndicator(ap); } #endif stack = (path_stack_entry *)TR; cl.Code = cl.CurrentCode = beg; sp = push_path(stack, NULL, &cl, &cint); add_to_index(&cint, first, sp, &cl); } static void contract_ftable(yamop *ipc, ClauseUnion *blk, PredEntry *ap, Functor f) { int n = ipc->y_u.sssl.s; FuncSwiEntry *fep; if (n > MIN_HASH_ENTRIES) { fep = lookup_f_hash(f, ipc->y_u.sssl.l, n); } else { fep = (FuncSwiEntry *)(ipc->y_u.sssl.l); while (fep->Tag != f) fep++; } fep->u_f.labp = FAILCODE; } static void contract_ctable(yamop *ipc, ClauseUnion *blk, PredEntry *ap, Term at) { int n = ipc->y_u.sssl.s; AtomSwiEntry *cep; if (n > MIN_HASH_ENTRIES) { cep = lookup_c_hash(at, ipc->y_u.sssl.l, n); } else { cep = (AtomSwiEntry *)(ipc->y_u.sssl.l); while (cep->Tag != at) cep++; } cep->u_a.labp = FAILCODE; } static void remove_from_index(PredEntry *ap, path_stack_entry *sp, ClauseDef *cls, yamop *bg, yamop *lt, struct intermediates *cint) { /* last clause to experiment with */ yamop *ipc = ap->cs.p_code.TrueCodeOfPred; if (ap->cs.p_code.NOfClauses == 1) { if (ap->PredFlags & IndexedPredFlag) { Yap_RemoveIndexation(ap); return; } ap->cs.p_code.TrueCodeOfPred = ap->cs.p_code.FirstClause; if (ap->PredFlags & (SpiedPredFlag | CountPredFlag | ProfiledPredFlag)) { ap->OpcodeOfPred = Yap_opcode(_spy_pred); ap->CodeOfPred = (yamop *)(&(ap->OpcodeOfPred)); #if defined(YAPOR) || defined(THREADS) } else if (ap->PredFlags & LogUpdatePredFlag && !(ap->PredFlags & ThreadLocalPredFlag) && ap->ModuleOfPred != IDB_MODULE) { ap->cs.p_code.TrueCodeOfPred = FAILCODE; ap->OpcodeOfPred = LOCKPRED_OPCODE; ap->CodeOfPred = (yamop *)(&(ap->OpcodeOfPred)); #endif } else { ap->OpcodeOfPred = ap->cs.p_code.FirstClause->opc; ap->CodeOfPred = ap->cs.p_code.TrueCodeOfPred; } return; } sp = init_block_stack(sp, ipc, ap); /* try to refine the interval using the indexing code */ while (ipc != NULL) { op_numbers op = Yap_op_from_opcode(ipc->opc); switch (op) { case _retry_profiled: case _count_retry: ipc = NEXTOP(ipc, p); break; case _try_in: /* I cannot expand a predicate that starts on a variable, have to expand the index. */ if (IN_BETWEEN(bg, ipc->y_u.l.l, lt)) { sp = kill_clause(ipc, bg, lt, sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { /* just go to next instruction */ ipc = NEXTOP(ipc, l); } break; case _try_clause: case _retry: /* I cannot expand a predicate that starts on a variable, have to expand the index. */ if (IN_BETWEEN(bg, ipc->y_u.Otapl.d, lt)) { sp = kill_clause(ipc, bg, lt, sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { /* just go to next instruction */ ipc = NEXTOP(ipc, Otapl); } break; case _try_clause2: case _try_clause3: case _try_clause4: case _retry2: case _retry3: case _retry4: /* I cannot expand a predicate that starts on a variable, have to expand the index. */ if (IN_BETWEEN(bg, ipc->y_u.l.l, lt)) { sp = kill_clause(ipc, bg, lt, sp, ap); ipc = pop_path(&sp, cls, ap, cint); } else { /* just go to next instruction */ ipc = NEXTOP(ipc, l); } break; case _trust: if (IN_BETWEEN(bg, ipc->y_u.Otapl.d, lt)) { sp = kill_clause(ipc, bg, lt, sp, ap); } ipc = pop_path(&sp, cls, ap, cint); break; case _enter_lu_pred: ipc->y_u.Illss.s--; ipc->y_u.Illss.e++; #if DEBUG Yap_DirtyCps++; Yap_LiveCps--; #endif sp = kill_clause(ipc, bg, lt, sp, ap); ipc = pop_path(&sp, cls, ap, cint); break; /* instructions type l */ case _try_me: case _retry_me: sp = push_path(sp, &(ipc->y_u.Otapl.d), cls, cint); ipc = NEXTOP(ipc, Otapl); break; case _profiled_trust_me: case _trust_me: case _count_trust_me: ipc = NEXTOP(ipc, Otapl); break; case _jump: sp = cross_block(sp, &ipc->y_u.l.l, ap, cint); /* just skip for now, but should worry about memory management */ ipc = ipc->y_u.l.l; break; case _jump_if_var: sp = push_path(sp, &(ipc->y_u.l.l), cls, cint); ipc = NEXTOP(ipc, l); break; case _jump_if_nonvar: sp = push_path(sp, &(ipc->y_u.xll.l2), cls, cint); sp = cross_block(sp, &ipc->y_u.xll.l1, ap, cint); ipc = ipc->y_u.xll.l1; break; case _user_switch: ipc = ipc->y_u.lp.l; break; /* instructions type e */ case _switch_on_type: sp = push_path(sp, &(ipc->y_u.llll.l4), cls, cint); if (ap->PredFlags & LogUpdatePredFlag) { add_head_info(cls, 1); } else { add_info(cls, 1); } if (IsPairTerm(cls->Tag)) { yamop *nipc = ipc->y_u.llll.l1; if (IN_BETWEEN(bg, nipc, lt)) { /* jump straight to clause */ ipc->y_u.llll.l1 = FAILCODE; ipc = pop_path(&sp, cls, ap, cint); } else { /* go on */ sp = cross_block(sp, &ipc->y_u.llll.l1, ap, cint); ipc = nipc; } } else if (IsAtomOrIntTerm(cls->Tag)) { yamop *nipc = ipc->y_u.llll.l2; if (IN_BETWEEN(bg, nipc, lt)) { /* jump straight to clause */ ipc->y_u.llll.l2 = FAILCODE; ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else if (IsApplTerm(cls->Tag)) { yamop *nipc = ipc->y_u.llll.l3; if (IN_BETWEEN(bg, nipc, lt)) { /* jump straight to clause */ ipc->y_u.llll.l3 = FAILCODE; ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else { /* we can't separate into four groups, need to restart. */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } break; case _switch_list_nl: sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); break; case _switch_on_arg_type: sp = push_path(sp, &(ipc->y_u.xllll.l4), cls, cint); if (ap->PredFlags & LogUpdatePredFlag) { add_head_info(cls, Yap_regtoregno(ipc->y_u.xllll.x)); } else { add_info(cls, Yap_regtoregno(ipc->y_u.xllll.x)); } if (IsPairTerm(cls->Tag)) { yamop *nipc = ipc->y_u.xllll.l1; if (IN_BETWEEN(bg, nipc, lt)) { /* jump straight to clause */ ipc->y_u.xllll.l1 = FAILCODE; ipc = pop_path(&sp, cls, ap, cint); } else { /* go on */ sp = cross_block(sp, &ipc->y_u.xllll.l1, ap, cint); ipc = nipc; } } else if (IsAtomOrIntTerm(cls->Tag)) { yamop *nipc = ipc->y_u.xllll.l2; if (IN_BETWEEN(bg, nipc, lt)) { /* jump straight to clause */ ipc->y_u.xllll.l2 = FAILCODE; ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else if (IsApplTerm(cls->Tag)) { yamop *nipc = ipc->y_u.xllll.l3; if (IN_BETWEEN(bg, nipc, lt)) { /* jump straight to clause */ ipc->y_u.xllll.l3 = FAILCODE; ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else { /* we can't separate into four groups, need to restart. */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } break; case _switch_on_sub_arg_type: sp = push_path(sp, &(ipc->y_u.sllll.l4), cls, cint); add_arg_info(cls, ap, ipc->y_u.sllll.s + 1); if (IsPairTerm(cls->Tag)) { yamop *nipc = ipc->y_u.sllll.l1; if (IN_BETWEEN(bg, nipc, lt)) { /* jump straight to clause */ ipc->y_u.sllll.l1 = FAILCODE; ipc = pop_path(&sp, cls, ap, cint); } else { /* go on */ sp = cross_block(sp, &ipc->y_u.sllll.l1, ap, cint); ipc = nipc; } } else if (IsAtomOrIntTerm(cls->Tag)) { yamop *nipc = ipc->y_u.sllll.l2; if (IN_BETWEEN(bg, nipc, lt)) { /* jump straight to clause */ ipc->y_u.sllll.l2 = FAILCODE; ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else if (IsApplTerm(cls->Tag)) { yamop *nipc = ipc->y_u.sllll.l3; if (IN_BETWEEN(bg, nipc, lt)) { /* jump straight to clause */ ipc->y_u.sllll.l3 = FAILCODE; ipc = pop_path(&sp, cls, ap, cint); } else { /* I do not have to worry about crossing a block here */ ipc = nipc; } } else { /* we can't separate into four groups, need to restart. */ sp = kill_block(sp, ap); ipc = pop_path(&sp, cls, ap, cint); } break; case _if_not_then: ipc = pop_path(&sp, cls, ap, cint); break; /* instructions type ollll */ case _switch_on_func: case _if_func: case _go_on_func: { FuncSwiEntry *fe; yamop *newpc; Functor f = (Functor)RepAppl(cls->Tag); if (op == _switch_on_func) { fe = lookup_f_hash(f, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } else { fe = lookup_f(f, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } newpc = fe->u_f.labp; if (newpc == (yamop *)&(ap->cs.p_code.ExpandCode)) { /* we found it */ ipc = pop_path(&sp, cls, ap, cint); } else if (newpc == FAILCODE) { ipc = pop_path(&sp, cls, ap, cint); } else if (IN_BETWEEN(bg, fe->u_f.Label, lt)) { /* oops, nothing there */ contract_ftable(ipc, current_block(sp), ap, f); ipc = pop_path(&sp, cls, ap, cint); } else { yamop *newpc = fe->u_f.labp; sp = fetch_new_block(sp, &(ipc->y_u.sssl.l), ap, cint); sp = cross_block(sp, &(fe->u_f.labp), ap, cint); ipc = newpc; } } break; case _index_dbref: cls->Tag = cls->ucd.t_ptr; ipc = NEXTOP(ipc, e); break; case _index_blob: cls->Tag = Yap_Double_key(cls->ucd.t_ptr); ipc = NEXTOP(ipc, e); break; case _index_long: cls->Tag = Yap_Int_key(cls->ucd.t_ptr); ipc = NEXTOP(ipc, e); break; case _switch_on_cons: case _if_cons: case _go_on_cons: { AtomSwiEntry *ae; yamop *newpc; Term at = cls->Tag; if (op == _switch_on_cons) { ae = lookup_c_hash(at, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } else { ae = lookup_c(at, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } newpc = ae->u_a.labp; if (newpc == (yamop *)&(ap->cs.p_code.ExpandCode)) { /* we found it */ ipc = pop_path(&sp, cls, ap, cint); } else if (newpc == FAILCODE) { ipc = pop_path(&sp, cls, ap, cint); } else if (IN_BETWEEN(bg, ae->u_a.Label, lt)) { /* oops, nothing there */ contract_ctable(ipc, current_block(sp), ap, at); ipc = pop_path(&sp, cls, ap, cint); } else { yamop *newpc = ae->u_a.labp; sp = fetch_new_block(sp, &(ipc->y_u.sssl.l), ap, cint); sp = cross_block(sp, &(ae->u_a.labp), ap, cint); ipc = newpc; } } break; case _expand_index: ipc = pop_path(&sp, cls, ap, cint); break; case _expand_clauses: nullify_expand_clause(ipc, sp, cls); ipc = pop_path(&sp, cls, ap, cint); break; case _lock_lu: ipc = NEXTOP(ipc, p); break; default: if (IN_BETWEEN(bg, ipc, lt)) { sp = kill_unsafe_block(sp, op, ap, TRUE, TRUE, cls); } ipc = pop_path(&sp, cls, ap, cint); } } } /* clause is locked */ void Yap_RemoveClauseFromIndex(PredEntry *ap, yamop *beg) { CACHE_REGS ClauseDef cl; /* first clause */ path_stack_entry *stack, *sp; int cb; yamop *last; struct intermediates cint; if (ap->PredFlags & MegaClausePredFlag) { return; } cint.expand_block = NULL; cint.CodeStart = cint.BlobsStart = cint.cpc = cint.icpc = NULL; if ((cb = sigsetjmp(cint.CompilerBotch, 0)) == 3) { restore_machine_regs(); Yap_gcl(LOCAL_Error_Size, ap->ArityOfPE, ENV, CP); save_machine_regs(); } else if (cb == 2) { restore_machine_regs(); Yap_growheap(FALSE, LOCAL_Error_Size, NULL); save_machine_regs(); } else if (cb == 4) { restore_machine_regs(); Yap_growtrail(LOCAL_Error_Size, FALSE); save_machine_regs(); } LOCAL_Error_Size = 0; LOCAL_ErrorMessage = NULL; cint.term_depth = cint.last_index_new_depth = cint.last_depth_size = 0L; if (cb || (ap->cs.p_code.NOfClauses == 2 && ap->PredFlags & IndexedPredFlag)) { /* cannot rely on the code */ if (ap->PredFlags & LogUpdatePredFlag) { Yap_kill_iblock( (ClauseUnion *)ClauseCodeToLogUpdIndex(ap->cs.p_code.TrueCodeOfPred), NULL, ap); } else { StaticIndex *cl; ap->PredFlags &= ~LogUpdatePredFlag; cl = ClauseCodeToStaticIndex(ap->cs.p_code.TrueCodeOfPred); Yap_kill_iblock((ClauseUnion *)cl, NULL, ap); } ap->PredFlags &= ~IndexedPredFlag; return; } #if DEBUG if (GLOBAL_Option['i' - 'a' + 1]) { Term tmod = ap->ModuleOfPred; if (!tmod) tmod = TermProlog; Yap_DebugPutc(stderr, '-'); Yap_DebugPutc(stderr, '\t'); Yap_DebugPlWrite(tmod); Yap_DebugPutc(stderr, ':'); if (ap->ModuleOfPred != IDB_MODULE) { if (ap->ArityOfPE == 0) { Atom At = (Atom)ap->FunctorOfPred; Yap_DebugPlWrite(MkAtomTerm(At)); } else { Functor f = ap->FunctorOfPred; Atom At = NameOfFunctor(f); Yap_DebugPlWrite(MkAtomTerm(At)); Yap_DebugPutc(stderr, '/'); Yap_DebugPlWrite(MkIntegerTerm(ArityOfFunctor(f))); } } else { if (ap->PredFlags & NumberDBPredFlag) { Int id = ap->src.IndxId; Yap_DebugPlWrite(MkIntegerTerm(id)); } else if (ap->PredFlags & AtomDBPredFlag) { Atom At = (Atom)ap->FunctorOfPred; Yap_DebugPlWrite(MkAtomTerm(At)); } else { Functor f = ap->FunctorOfPred; Atom At = NameOfFunctor(f); Yap_DebugPlWrite(MkAtomTerm(At)); Yap_DebugPutc(stderr, '/'); Yap_DebugPlWrite(MkIntegerTerm(ArityOfFunctor(f))); } } Yap_DebugPutc(stderr, '\n'); } #endif stack = (path_stack_entry *)TR; if (ap->PredFlags & LogUpdatePredFlag) { LogUpdClause *c = ClauseCodeToLogUpdClause(beg); cl.Code = cl.CurrentCode = beg; last = (yamop *)((CODEADDR)c + c->ClSize); } else { StaticClause *c = ClauseCodeToStaticClause(beg); cl.Code = cl.CurrentCode = beg; last = (yamop *)((CODEADDR)c + c->ClSize); } sp = push_path(stack, NULL, &cl, &cint); if (ap->cs.p_code.NOfClauses == 0) { /* there was no indexing code */ #if defined(YAPOR) || defined(THREADS) if (ap->PredFlags & LogUpdatePredFlag && ap->ModuleOfPred != IDB_MODULE) { ap->cs.p_code.TrueCodeOfPred = FAILCODE; ap->CodeOfPred = (yamop *)(&(ap->OpcodeOfPred)); } else { #endif ap->CodeOfPred = ap->cs.p_code.TrueCodeOfPred = FAILCODE; #if defined(YAPOR) || defined(THREADS) } #endif ap->OpcodeOfPred = Yap_opcode(_op_fail); } else if (ap->PredFlags & IndexedPredFlag) { remove_from_index(ap, sp, &cl, beg, last, &cint); } else if (ap->cs.p_code.NOfClauses == 1) { ap->cs.p_code.TrueCodeOfPred = ap->cs.p_code.FirstClause; ap->CodeOfPred = (yamop *)(&(ap->OpcodeOfPred)); } } static void store_clause_choice_point(Term t1, Term tb, Term tr, yamop *ipc, PredEntry *pe, yamop *ap_pc, yamop *cp_pc USES_REGS) { Term tpc = MkIntegerTerm((Int)ipc); Term tpe = MkIntegerTerm((Int)pe); CELL *tsp = ASP - 5; choiceptr bptr = ((choiceptr)tsp) - 1; tsp[0] = tpe; tsp[1] = tpc; tsp[2] = t1; tsp[3] = tb; tsp[4] = tr; bptr->cp_tr = TR; HB = bptr->cp_h = HR; #ifdef DEPTH_LIMIT bptr->cp_depth = DEPTH; #endif bptr->cp_b = B; bptr->cp_cp = cp_pc; bptr->cp_ap = ap_pc; bptr->cp_env = ENV; /* now, install the new YREG */ ASP = (CELL *)bptr; ASP[E_CB] = (CELL)bptr; B = bptr; #ifdef YAPOR SCH_set_load(B); #endif /* YAPOR */ SET_BB(bptr); } static void update_clause_choice_point(yamop *ipc, yamop *ap_pc USES_REGS) { Term tpc = MkIntegerTerm((Int)ipc); B->cp_args[1] = tpc; B->cp_h = HR; B->cp_ap = ap_pc; } static LogUpdClause *to_clause(yamop *ipc, PredEntry *ap) { if (ap->PredFlags & LogUpdatePredFlag) return lu_clause(ipc, ap); else if (ap->PredFlags & MegaClausePredFlag) return (LogUpdClause *)ipc; else return (LogUpdClause *)simple_static_clause(ipc, ap); } LogUpdClause *Yap_FollowIndexingCode(PredEntry *ap, yamop *ipc, Term Terms[3], yamop *ap_pc, yamop *cp_pc) { CACHE_REGS CELL *s_reg = NULL; Term t = TermNil; int blob_term = FALSE; choiceptr b0 = NULL; #if defined(YAPOR) || defined(THREADS) yamop **jlbl = NULL; #endif pred_flags_t lu_pred = ap->PredFlags & LogUpdatePredFlag; int unbounded = TRUE; if (ap->ModuleOfPred != IDB_MODULE) { if (ap->ArityOfPE) { CELL *tar = RepAppl(Deref(Terms[0])); UInt i; for (i = 1; i <= ap->ArityOfPE; i++) { XREGS[i] = tar[i]; } } } /* try to refine the interval using the indexing code */ while (ipc != NULL) { op_numbers op = Yap_op_from_opcode(ipc->opc); switch (op) { case _try_in: update_clause_choice_point(NEXTOP(ipc, l), ap_pc PASS_REGS); if (lu_pred) return lu_clause(ipc->y_u.l.l, ap); else return (LogUpdClause *)static_clause(ipc->y_u.l.l, ap, unbounded); break; case _try_clause: #if TABLING case _table_try: #endif if (b0 == NULL) store_clause_choice_point(Terms[0], Terms[1], Terms[2], NEXTOP(ipc, Otapl), ap, ap_pc, cp_pc PASS_REGS); else { B = b0; b0 = NULL; update_clause_choice_point(NEXTOP(ipc, Otapl), ap_pc PASS_REGS); } if (lu_pred) return lu_clause(ipc->y_u.Otapl.d, ap); else return (LogUpdClause *)static_clause(ipc->y_u.Otapl.d, ap, unbounded); case _try_clause2: case _try_clause3: case _try_clause4: if (b0 == NULL) store_clause_choice_point(Terms[0], Terms[1], Terms[2], NEXTOP(ipc, l), ap, ap_pc, cp_pc PASS_REGS); else { B = b0; b0 = NULL; update_clause_choice_point(NEXTOP(ipc, l), ap_pc PASS_REGS); } if (lu_pred) return lu_clause(ipc->y_u.l.l, ap); else return (LogUpdClause *)static_clause(ipc->y_u.l.l, ap, unbounded); case _try_me: #if TABLING case _table_try_me: #endif if (b0 == NULL) store_clause_choice_point(Terms[0], Terms[1], Terms[2], ipc->y_u.Otapl.d, ap, ap_pc, cp_pc PASS_REGS); else { B = b0; b0 = NULL; update_clause_choice_point(ipc->y_u.Otapl.d, ap_pc PASS_REGS); } ipc = NEXTOP(ipc, Otapl); break; case _retry_profiled: case _count_retry: ipc = NEXTOP(ipc, p); break; case _retry: #if TABLING case _table_retry: #endif update_clause_choice_point(NEXTOP(ipc, Otapl), ap_pc PASS_REGS); if (lu_pred) return lu_clause(ipc->y_u.Otapl.d, ap); else return (LogUpdClause *)static_clause(ipc->y_u.Otapl.d, ap, TRUE); case _retry2: case _retry3: case _retry4: update_clause_choice_point(NEXTOP(ipc, l), ap_pc PASS_REGS); if (lu_pred) return lu_clause(ipc->y_u.l.l, ap); else return (LogUpdClause *)static_clause(ipc->y_u.l.l, ap, TRUE); case _retry_me: update_clause_choice_point(ipc->y_u.Otapl.d, ap_pc PASS_REGS); ipc = NEXTOP(ipc, Otapl); break; case _trust: #if TABLING case _table_trust: #endif { while (POP_CHOICE_POINT(B->cp_b)) { POP_EXECUTE(); } } #ifdef YAPOR { choiceptr cut_pt; cut_pt = B->cp_b; CUT_prune_to(cut_pt); B = cut_pt; } #else B = B->cp_b; #endif /* YAPOR */ b0 = B; if (lu_pred) return lu_clause(ipc->y_u.Otapl.d, ap); else return (LogUpdClause *)static_clause(ipc->y_u.Otapl.d, ap, TRUE); case _profiled_trust_me: case _trust_me: case _count_trust_me: #if TABLING case _table_trust_me: #endif b0 = B; { while (POP_CHOICE_POINT(B->cp_b)) { POP_EXECUTE(); } } #ifdef YAPOR { choiceptr cut_pt; cut_pt = B->cp_b; CUT_prune_to(cut_pt); B = cut_pt; } #else B = B->cp_b; #endif /* YAPOR */ ipc = NEXTOP(ipc, Otapl); break; case _enter_lu_pred: { LogUpdIndex *cl = ipc->y_u.Illss.I; PredEntry *ap = cl->ClPred; if (!cl) return NULL; /* in case the index is empty */ if (ap->LastCallOfPred != LUCALL_EXEC) { /* only increment time stamp if we are working on current time stamp */ if (ap->TimeStampOfPred >= TIMESTAMP_RESET) Yap_UpdateTimestamps(ap); ap->TimeStampOfPred++; /* fprintf(stderr,"R * %x--%d--%ul\n",ap,ap->TimeStampOfPred,ap->ArityOfPE);*/ ap->LastCallOfPred = LUCALL_EXEC; } *--ASP = MkIntegerTerm(ap->TimeStampOfPred); /* indicate the indexing code is being used */ #if MULTIPLE_STACKS /* just store a reference */ INC_CLREF_COUNT(cl); TRAIL_CLREF(cl); #else if (!(cl->ClFlags & InUseMask)) { cl->ClFlags |= InUseMask; TRAIL_CLREF(cl); } #endif } ipc = ipc->y_u.Illss.l1; break; case _try_logical: if (b0 == NULL) store_clause_choice_point(Terms[0], Terms[1], Terms[2], ipc->y_u.OtaLl.n, ap, ap_pc, cp_pc PASS_REGS); else { B = b0; b0 = NULL; update_clause_choice_point(ipc->y_u.OtaLl.n, ap_pc PASS_REGS); } { UInt timestamp = IntegerOfTerm(((CELL *)(B + 1))[5]); if (!VALID_TIMESTAMP(timestamp, ipc->y_u.OtaLl.d)) { /* jump to next instruction */ ipc = ipc->y_u.OtaLl.n; break; } } return ipc->y_u.OtaLl.d; case _retry_logical: case _profiled_retry_logical: case _count_retry_logical: { UInt timestamp = IntegerOfTerm(((CELL *)(B + 1))[5]); if (!VALID_TIMESTAMP(timestamp, ipc->y_u.OtaLl.d)) { /* jump to next instruction */ ipc = ipc->y_u.OtaLl.n; break; } } update_clause_choice_point(ipc->y_u.OtaLl.n, ap_pc PASS_REGS); return ipc->y_u.OtaLl.d; #if TABLING case _table_try_single: return (LogUpdClause *)ClauseCodeToStaticClause(ipc); #endif case _trust_logical: case _count_trust_logical: case _profiled_trust_logical: { UInt timestamp = IntegerOfTerm(((CELL *)(B + 1))[5]); LogUpdIndex *cl = ipc->y_u.OtILl.block; LogUpdClause *newpc; if (!VALID_TIMESTAMP(timestamp, ipc->y_u.OtILl.d)) { /* jump to next instruction */ newpc = NULL; } else { newpc = ipc->y_u.OtILl.d; } #if MULTIPLE_STACKS DEC_CLREF_COUNT(cl); B->cp_tr--; TR--; /* actually get rid of the code */ if (cl->ClRefCount == 0 && cl->ClFlags & (ErasedMask | DirtyMask)) { /* I am the last one using this clause, hence I don't need a lock to dispose of it. But on the other hand I need to make sure the clause is still there when I am back. */ LogUpdClause *lcl = ipc->y_u.OtILl.d; if (newpc) { if (lcl->ClRefCount == 1) { /* make sure the clause isn't destroyed */ /* always add an extra reference */ INC_CLREF_COUNT(lcl); TRAIL_CLREF(lcl); B->cp_tr = TR; } } if (cl->ClFlags & ErasedMask) { Yap_ErLogUpdIndex(cl); } else { Yap_CleanUpIndex(cl); } } #else if (TrailTerm(B->cp_tr - 1) == CLREF_TO_TRENTRY(cl) && B->cp_tr != B->cp_b->cp_tr) { B->cp_tr--; TR--; cl->ClFlags &= ~InUseMask; /* next, recover space for the indexing code if it was erased */ if (cl->ClFlags & (ErasedMask | DirtyMask)) { LogUpdClause *lcl = ipc->y_u.OtILl.d; /* make sure we don't erase the clause we are jumping to, notice that ErLogUpdIndex may remove several references in one go. Notice we only need to do this if we´ re jumping to the clause. */ if (newpc && !(lcl->ClFlags & (DirtyMask | InUseMask))) { lcl->ClFlags |= InUseMask; TRAIL_CLREF(lcl); } if (cl->ClFlags & ErasedMask) { Yap_ErLogUpdIndex(cl); } else { Yap_CleanUpIndex(cl); } } } #endif { while (POP_CHOICE_POINT(B->cp_b)) { POP_EXECUTE(); } } #ifdef YAPOR { choiceptr cut_pt; cut_pt = B->cp_b; CUT_prune_to(cut_pt); B = cut_pt; } #else B = B->cp_b; #endif /* YAPOR */ b0 = B; return newpc; } case _jump: ipc = ipc->y_u.l.l; break; case _jump_if_var: { Term t = Deref(ARG1); if (IsVarTerm(t)) { SET_JLBL(l.l); ipc = ipc->y_u.l.l; } else { ipc = NEXTOP(ipc, l); } } break; case _jump_if_nonvar: { Term t = Deref(XREGS[arg_from_x(ipc->y_u.xll.x)]); if (!IsVarTerm(t)) { SET_JLBL(xll.l1); ipc = ipc->y_u.xll.l1; } else { ipc = NEXTOP(ipc, xll); } } break; case _user_switch: ipc = ipc->y_u.lp.l; break; /* instructions type e */ case _switch_on_type: t = Deref(ARG1); blob_term = FALSE; if (IsVarTerm(t)) { SET_JLBL(llll.l4); ipc = ipc->y_u.llll.l4; } else if (IsPairTerm(t)) { unbounded = FALSE; SET_JLBL(llll.l1); ipc = ipc->y_u.llll.l1; S = s_reg = RepPair(t); } else if (IsAtomOrIntTerm(t)) { SET_JLBL(llll.l2); ipc = ipc->y_u.llll.l2; } else { SET_JLBL(llll.l3); ipc = ipc->y_u.llll.l3; S = RepAppl(t); } break; case _switch_list_nl: t = Deref(ARG1); blob_term = FALSE; if (IsVarTerm(t)) { SET_JLBL(ollll.l4); ipc = ipc->y_u.ollll.l4; } else if (IsPairTerm(t)) { unbounded = FALSE; SET_JLBL(ollll.l1); ipc = ipc->y_u.ollll.l1; S = s_reg = RepPair(t); } else if (t == TermNil) { unbounded = FALSE; SET_JLBL(ollll.l2); ipc = ipc->y_u.ollll.l2; } else { SET_JLBL(ollll.l3); ipc = ipc->y_u.ollll.l3; S = RepAppl(t); } break; case _switch_on_arg_type: t = Deref(XREGS[arg_from_x(ipc->y_u.xllll.x)]); blob_term = FALSE; if (IsVarTerm(t)) { SET_JLBL(xllll.l4); ipc = ipc->y_u.xllll.l4; } else if (IsPairTerm(t)) { unbounded = FALSE; SET_JLBL(xllll.l1); ipc = ipc->y_u.xllll.l1; S = s_reg = RepPair(t); } else if (IsAtomOrIntTerm(t)) { SET_JLBL(xllll.l2); ipc = ipc->y_u.xllll.l2; } else { SET_JLBL(xllll.l3); ipc = ipc->y_u.xllll.l3; S = RepAppl(t); } break; case _switch_on_sub_arg_type: t = Deref(s_reg[ipc->y_u.sllll.s]); blob_term = FALSE; if (IsVarTerm(t)) { SET_JLBL(sllll.l4); ipc = ipc->y_u.sllll.l4; } else if (IsPairTerm(t)) { unbounded = FALSE; SET_JLBL(sllll.l1); S = s_reg = RepPair(t); ipc = ipc->y_u.sllll.l1; } else if (IsAtomOrIntTerm(t)) { SET_JLBL(sllll.l2); ipc = ipc->y_u.sllll.l2; } else { SET_JLBL(sllll.l3); ipc = ipc->y_u.sllll.l3; S = RepAppl(t); } break; case _if_not_then: t = Deref(ARG1); blob_term = FALSE; if (IsVarTerm(t)) { SET_JLBL(clll.l3); ipc = ipc->y_u.clll.l3; } else if (!IsVarTerm(t) && t != ipc->y_u.clll.c) { SET_JLBL(clll.l1); ipc = ipc->y_u.clll.l1; } else { SET_JLBL(clll.l2); ipc = ipc->y_u.clll.l2; } break; /* instructions type ollll */ case _switch_on_func: case _if_func: case _go_on_func: { FuncSwiEntry *fe; Functor f; unbounded = FALSE; s_reg = RepAppl(t); f = (Functor)s_reg[0]; s_reg++; S = s_reg; if (op == _switch_on_func) { fe = lookup_f_hash(f, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } else { fe = lookup_f(f, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } #if defined(YAPOR) || defined(THREADS) jlbl = &(fe->u_f.labp); #endif ipc = fe->u_f.labp; } break; case _index_dbref: if (s_reg[-1] != (CELL)FunctorDBREF) { ipc = FAILCODE; break; } t = AbsAppl(s_reg - 1); blob_term = FALSE; ipc = NEXTOP(ipc, e); break; case _index_blob: if (s_reg[-1] != (CELL)FunctorDouble) { ipc = FAILCODE; break; } t = Yap_DoubleP_key(s_reg); blob_term = TRUE; ipc = NEXTOP(ipc, e); break; case _index_long: if (s_reg[-1] != (CELL)FunctorLongInt) { ipc = FAILCODE; break; } t = Yap_IntP_key(s_reg); blob_term = TRUE; ipc = NEXTOP(ipc, e); break; case _switch_on_cons: case _if_cons: case _go_on_cons: { AtomSwiEntry *ae; unbounded = FALSE; if (op == _switch_on_cons) { ae = lookup_c_hash(t, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } else { ae = lookup_c(t, ipc->y_u.sssl.l, ipc->y_u.sssl.s); } #if defined(YAPOR) || defined(THREADS) jlbl = &(ae->u_a.labp); #endif ipc = ae->u_a.labp; } break; case _expand_index: case _expand_clauses: if (blob_term) { /* protect garbage collector */ XREGS[ap->ArityOfPE + 1] = (CELL)&XREGS[ap->ArityOfPE + 1]; XREGS[ap->ArityOfPE + 2] = TermNil; } else { XREGS[ap->ArityOfPE + 1] = (CELL)s_reg; XREGS[ap->ArityOfPE + 2] = t; } XREGS[ap->ArityOfPE + 3] = Terms[0]; XREGS[ap->ArityOfPE + 4] = Terms[1]; XREGS[ap->ArityOfPE + 5] = Terms[2]; #if defined(YAPOR) || defined(THREADS) if (!same_lu_block(jlbl, ipc)) { ipc = *jlbl; break; } #endif ipc = ExpandIndex(ap, 5, cp_pc PASS_REGS); if (!blob_term) { /* protect garbage collector */ s_reg = (CELL *)XREGS[ap->ArityOfPE + 1]; t = XREGS[ap->ArityOfPE + 2]; } blob_term = FALSE; Terms[0] = XREGS[ap->ArityOfPE + 3]; Terms[1] = XREGS[ap->ArityOfPE + 4]; Terms[2] = XREGS[ap->ArityOfPE + 5]; break; case _undef_p: return NULL; case _lock_lu: ipc = NEXTOP(ipc, p); break; #if THREADS case _thread_local: ap = Yap_GetThreadPred(ap PASS_REGS); ipc = ap->CodeOfPred; break; #endif case _spy_pred: case _lock_pred: if ((ap->PredFlags & IndexedPredFlag) || ap->cs.p_code.NOfClauses <= 1) { ipc = ap->cs.p_code.TrueCodeOfPred; break; } case _index_pred: if (blob_term) { /* protect garbage collector */ XREGS[ap->ArityOfPE + 1] = (CELL)&XREGS[ap->ArityOfPE + 1]; XREGS[ap->ArityOfPE + 2] = TermNil; } else { XREGS[ap->ArityOfPE + 1] = (CELL)s_reg; XREGS[ap->ArityOfPE + 2] = t; } XREGS[ap->ArityOfPE + 3] = Terms[0]; XREGS[ap->ArityOfPE + 4] = Terms[1]; XREGS[ap->ArityOfPE + 5] = Terms[2]; Yap_IPred(ap, 5, cp_pc); ipc = ap->cs.p_code.TrueCodeOfPred; if (!blob_term) { /* protect garbage collector */ s_reg = (CELL *)XREGS[ap->ArityOfPE + 1]; t = XREGS[ap->ArityOfPE + 2]; } Terms[0] = XREGS[ap->ArityOfPE + 3]; Terms[1] = XREGS[ap->ArityOfPE + 4]; Terms[2] = XREGS[ap->ArityOfPE + 5]; break; case _op_fail: if (ipc == FAILCODE) return NULL; default: if (b0) { { while (POP_CHOICE_POINT(B->cp_b)) { POP_EXECUTE(); } } #ifdef YAPOR { choiceptr cut_pt; cut_pt = B->cp_b; CUT_prune_to(cut_pt); B = cut_pt; } #else B = B->cp_b; #endif /* YAPOR */ /* I did a trust */ } if (op == _op_fail) return NULL; if (lu_pred) return lu_clause(ipc, ap); else return (LogUpdClause *)static_clause(ipc, ap, unbounded); } } if (b0) { /* I did a trust */ { while (POP_CHOICE_POINT(B->cp_b)) { POP_EXECUTE(); } } #ifdef YAPOR { choiceptr cut_pt; cut_pt = B->cp_b; CUT_prune_to(cut_pt); B = cut_pt; } #else B = B->cp_b; #endif /* YAPOR */ } return NULL; } LogUpdClause *Yap_NthClause(PredEntry *ap, Int ncls) { CACHE_REGS yamop *ipc = ap->cs.p_code.TrueCodeOfPred, *alt = NULL; #if defined(YAPOR) || defined(THREADS) yamop **jlbl = NULL; #endif /* search every clause */ if (ncls > ap->cs.p_code.NOfClauses) return NULL; else if (ncls == 1) return to_clause(ap->cs.p_code.FirstClause, ap); else if (ap->PredFlags & MegaClausePredFlag) { MegaClause *mcl = ClauseCodeToMegaClause(ap->cs.p_code.FirstClause); /* fast access to nth element, all have same size */ return (LogUpdClause *)((char *)mcl->ClCode + (ncls - 1) * mcl->ClItemSize); } else if (ncls == ap->cs.p_code.NOfClauses) { return to_clause(ap->cs.p_code.LastClause, ap); } else if (ncls < 0) return NULL; if (ap->ModuleOfPred != IDB_MODULE) { if (ap->ArityOfPE) { UInt i; for (i = 1; i <= ap->ArityOfPE; i++) { XREGS[i] = MkVarTerm(); } } } else { ARG2 = MkVarTerm(); } while (TRUE) { op_numbers op = Yap_op_from_opcode(ipc->opc); switch (op) { case _try_in: if (ncls == 1) return to_clause(ipc->y_u.l.l, ap); ncls--; ipc = NEXTOP(ipc, l); break; case _retry_profiled: case _count_retry: ipc = NEXTOP(ipc, p); case _try_clause: case _retry: if (ncls == 1) return to_clause(ipc->y_u.Otapl.d, ap); else if (alt == NULL) { ncls--; /* get there in a fell swoop */ if (ap->PredFlags & ProfiledPredFlag) { if (ap->PredFlags & CountPredFlag) { ipc = (yamop *)((char *)ipc + ncls * (UInt)NEXTOP( NEXTOP(NEXTOP((yamop *)NULL, Otapl), p), p)); } else { ipc = (yamop *)((char *)ipc + ncls * (UInt)NEXTOP(NEXTOP((yamop *)NULL, Otapl), p)); } } else if (ap->PredFlags & CountPredFlag) { ipc = (yamop *)((char *)ipc + ncls * (UInt)NEXTOP(NEXTOP((yamop *)NULL, Otapl), p)); } else { ipc = (yamop *)((char *)ipc + ncls * (UInt)NEXTOP((yamop *)NULL, Otapl)); } return to_clause(ipc->y_u.Otapl.d, ap); } else { ncls--; } ipc = NEXTOP(ipc, Otapl); break; case _try_clause2: case _try_clause3: case _try_clause4: case _retry2: case _retry3: case _retry4: if (ncls == 1) return to_clause(ipc->y_u.l.l, ap); else if (alt == NULL) { ncls--; /* get there in a fell swoop */ if (ap->PredFlags & ProfiledPredFlag) { if (ap->PredFlags & CountPredFlag) { ipc = (yamop *)((char *)ipc + ncls * (UInt)NEXTOP( NEXTOP(NEXTOP((yamop *)NULL, l), p), p)); } else { ipc = (yamop *)((char *)ipc + ncls * (UInt)NEXTOP(NEXTOP((yamop *)NULL, l), p)); } } else if (ap->PredFlags & CountPredFlag) { ipc = (yamop *)((char *)ipc + ncls * (UInt)NEXTOP(NEXTOP((yamop *)NULL, l), p)); } else { ipc = (yamop *)((char *)ipc + ncls * (UInt)NEXTOP((yamop *)NULL, l)); } return to_clause(ipc->y_u.l.l, ap); } else { ncls--; } ipc = NEXTOP(ipc, l); break; case _trust: if (ncls == 1) return to_clause(ipc->y_u.l.l, ap); ncls--; ipc = alt; break; case _try_me: case _retry_me: alt = ipc->y_u.Otapl.d; ipc = NEXTOP(ipc, Otapl); break; case _profiled_trust_me: case _trust_me: case _count_trust_me: alt = NULL; ipc = NEXTOP(ipc, Otapl); break; case _try_logical: case _retry_logical: case _count_retry_logical: case _profiled_retry_logical: if (VALID_TIMESTAMP(ap->TimeStampOfPred, ipc->y_u.OtaLl.d)) { if (ncls == 1) return ipc->y_u.OtaLl.d; ncls--; } ipc = ipc->y_u.OtaLl.n; break; case _trust_logical: case _count_trust_logical: case _profiled_trust_logical: if (VALID_TIMESTAMP(ap->TimeStampOfPred, ipc->y_u.OtILl.d)) { if (ncls == 1) return ipc->y_u.OtILl.d; } return NULL; case _enter_lu_pred: SET_JLBL(Illss.l1); ipc = ipc->y_u.Illss.l1; break; case _lock_lu: ipc = NEXTOP(ipc, p); break; case _jump: SET_JLBL(l.l); ipc = ipc->y_u.l.l; break; case _jump_if_var: SET_JLBL(l.l); ipc = ipc->y_u.l.l; break; case _jump_if_nonvar: ipc = NEXTOP(ipc, xll); break; case _user_switch: SET_JLBL(l.l); ipc = ipc->y_u.lp.l; break; /* instructions type e */ case _switch_on_type: SET_JLBL(llll.l4); ipc = ipc->y_u.llll.l4; break; case _switch_list_nl: SET_JLBL(ollll.l4); ipc = ipc->y_u.ollll.l4; break; case _switch_on_arg_type: SET_JLBL(xllll.l4); ipc = ipc->y_u.xllll.l4; break; case _switch_on_sub_arg_type: SET_JLBL(sllll.l4); ipc = ipc->y_u.sllll.l4; break; case _if_not_then: SET_JLBL(clll.l3); ipc = ipc->y_u.clll.l3; break; case _expand_index: case _expand_clauses: #if defined(YAPOR) || defined(THREADS) if (*jlbl != (yamop *)&(ap->cs.p_code.ExpandCode)) { ipc = *jlbl; break; } #endif ipc = ExpandIndex(ap, 0, CP PASS_REGS); break; case _op_fail: ipc = alt; break; case _lock_pred: case _index_pred: case _spy_pred: Yap_IPred(ap, 0, CP); ipc = ap->cs.p_code.TrueCodeOfPred; break; case _undef_p: default: return NULL; } } } void Yap_CleanUpIndex(LogUpdIndex *blk) { /* just compact the code */ yamop *start = blk->ClCode; op_numbers op = Yap_op_from_opcode(start->opc); blk->ClFlags &= ~DirtyMask; while (op == _lock_lu) { start = NEXTOP(start, p); op = Yap_op_from_opcode(start->opc); } while (op == _jump_if_nonvar) { start = NEXTOP(start, xll); op = Yap_op_from_opcode(start->opc); } remove_dirty_clauses_from_index(start); }