This repository has been archived on 2023-08-20. You can view files and clone it, but cannot push or open issues or pull requests.
yap-6.3/C/index.c

1426 lines
38 KiB
C
Raw Normal View History

/*************************************************************************
* *
* 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 *
* Last rev: 5/2/88 *
* mods: *
* comments: Indexing a Prolog predicate *
* *
*************************************************************************/
#ifdef SCCS
static char SccsId[] = "%W% %G%";
#endif
/*
* 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
*
*/
#include "Yap.h"
#include "compile.h"
#include "clause.h"
#include "index.h"
#ifdef YAPOR
#include "or.macros.h"
#endif /* YAPOR */
#ifdef DEBUG
#include "yapio.h"
#endif
#ifndef NULL
#define NULL (void *)0
#endif
int IPredArity;
STATIC_PROTO(int clause_has_cut, (yamop *));
STATIC_PROTO(int followed_by_cut, (yamop *));
STATIC_PROTO(void emit_tr, (compiler_vm_op, yamop *, int, int));
STATIC_PROTO(void emit_try, (compiler_vm_op, int, yamop *, int, int));
STATIC_PROTO(yamop * Body, (yamop *));
STATIC_PROTO(yamop * SecB, (yamop *));
STATIC_PROTO(yamop * SecLB, (yamop *));
STATIC_PROTO(yamop * ThiB, (yamop *));
STATIC_PROTO(yamop * ThiLB, (yamop *));
STATIC_PROTO(void emit_cp_inst, (compiler_vm_op, yamop *, int, int));
STATIC_PROTO(CELL emit_space, (compiler_vm_op, int, int));
STATIC_PROTO(CELL emit_go, (int, Term));
STATIC_PROTO(void emit_if_not, (Term, CELL, CELL));
STATIC_PROTO(void fill_go, (CELL *, CELL));
STATIC_PROTO(void fill_switch_slots, (CELL *, CELL, CELL, CELL, CELL, int));
STATIC_PROTO(void ClrSpace, (CELL *, int));
STATIC_PROTO(int NGroupsIn, (PredEntry *));
STATIC_PROTO(void CountGroups, (GroupDef *, int));
STATIC_PROTO(yamop *FindFirst, (int, int));
STATIC_PROTO(int SizeTable, (int));
STATIC_PROTO(void BuildHash, (CELL *, int, int, int));
STATIC_PROTO(void BuildIfTable, (CELL *, int));
STATIC_PROTO(void TreatEntry, (EntryDef *, int, int, GroupDef *));
STATIC_PROTO(CELL DealFixed, (ClauseDef *, int, compiler_vm_op, int, GroupDef *));
STATIC_PROTO(CELL DealFixedWithBips, (ClauseDef *, int, int, GroupDef *));
STATIC_PROTO(CELL DealCons, (int));
STATIC_PROTO(CELL DealAppl, (int));
STATIC_PROTO(CELL StartList, (int));
STATIC_PROTO(CELL DealLAt, (ClauseDef *, int, int));
STATIC_PROTO(CELL DealLAppl, (ClauseDef *, int, int));
STATIC_PROTO(CELL DealLList, (ClauseDef *, int, int, int));
STATIC_PROTO(int NoHeadVar, (ClauseDef *, int));
STATIC_PROTO(CELL DealList, (int));
STATIC_PROTO(CELL GetFailToGo, (int));
STATIC_PROTO(int IsExtendedSingle, (int));
STATIC_PROTO(int gr_has_cuts, (GroupDef *));
STATIC_PROTO(void EmitGrSwitch, (int));
STATIC_PROTO(int IndexNonVarGr, (int));
STATIC_PROTO(void IndexVarGr, (int));
STATIC_PROTO(int SimpleCase, (void));
STATIC_PROTO(int ComplexCase, (void));
STATIC_PROTO(int SpecialCases, (void));
static int NClauses, NGroups;
static ClauseDef *ArOfCl;
static GroupDef *Groups;
static EntryDef *Entries;
static CELL labelno;
static int ExtendedSingle, AtomsOnlyNil;
static yamop *FailAddress;
static CELL *StorePoint;
static yamop * FirstCl;
static int RemovedCl; /* There were some clauses removed */
static yamop * indexed_code_for_cut = NIL;
static CELL log_update;
static inline int
clause_has_cut(yamop * C)
{
#ifdef YAPOR
return(YAMOP_CUT(C));
#else
return(FALSE);
#endif /* YAPOR */
}
/*
Detect whether the next instruction is a cut.
Only useful for try and friends.
*/
static int followed_by_cut(yamop * code)
{
register yamop *p = code;
while (TRUE)
{
if (p->opc == opcode(_get_x_var))
p = NEXTOP(p,xx);
if (p->opc == opcode(_get_y_var))
p = NEXTOP(p,yx);
else if (p->opc == opcode(_allocate))
p = NEXTOP(p,e);
else if (p->opc == opcode(_unify_x_var))
p = NEXTOP(p,ox);
else if (p->opc == opcode(_unify_y_var))
p = NEXTOP(p,oy);
else if (p->opc == opcode(_unify_l_x_var))
p = NEXTOP(p,ox);
else if (p->opc == opcode(_unify_l_y_var))
p = NEXTOP(p,oy);
else if (p->opc == opcode(_unify_void))
p = NEXTOP(p,o);
else if (p->opc == opcode(_unify_n_voids))
p = NEXTOP(p,os);
else if (p->opc == opcode(_unify_l_void))
p = NEXTOP(p,o);
else if (p->opc == opcode(_unify_l_n_voids))
p = NEXTOP(p,os);
else if (p->opc == opcode(_cut))
return(TRUE);
else if (p->opc == opcode(_cut_t))
return(TRUE);
else if (p->opc == opcode(_cut_e))
return(TRUE);
else return(FALSE);
}
/* make lcc happy */
return(FALSE);
}
/* emits an opcode followed by an adress */
inline static void
emit_tr(compiler_vm_op op, yamop * Address, int NClauses, int HasCut)
{
emit(op, Unsigned(Address), (NClauses << 1) + HasCut);
}
/* emits an opcode followed by an adress */
static void
emit_try(compiler_vm_op op, int op_offset, yamop * Address, int NClauses, int HasCut)
{
if (op == try_op) {
/* if a try, then try to use cuts if possible */
indexed_code_for_cut = NIL;
if (followed_by_cut(Address)) {
indexed_code_for_cut = Address;
return;
}
}
emit((compiler_vm_op)((int)op + op_offset), Unsigned(Address), (NClauses << 1) + HasCut);
}
/*
* compute address of first instruction of clause: try addr,n
*/
inline static yamop *
Body(yamop * Arg)
{
return (NEXTOP(Arg,ld));
}
/*
* compute address of instruction after get_atom or get_struct try
* addr,n get_atom atom,reg
*/
inline static yamop *
SecB(yamop * Arg)
{
yamop *pc = NEXTOP(Arg,ld);
if (pc->opc == opcode(_get_struct))
return (NEXTOP(pc,xf));
else
return (NEXTOP(pc,xc));
}
/*
* compute address of instruction after get_list try addr,n
* et_list reg
*/
inline static yamop *
SecLB(yamop * Arg)
{
return (NEXTOP(NEXTOP(Arg,ld),x));
}
/*
* compute address of instruction after get_list and unify try
* addr,n get_list reg
* nify_atom atom
*/
inline static yamop *
ThiB(yamop * Arg)
{
yamop *pc = NEXTOP(NEXTOP(Arg,ld),x);
if (pc->opc == opcode(_unify_struct))
return (NEXTOP(pc,of));
else
return (NEXTOP(pc,oc));
}
/*
* compute address of instruction after get_list unify_list try
* addr,n get_list reg unify_list
*/
inline static yamop *
ThiLB(yamop * Arg)
{
return ( NEXTOP(NEXTOP(NEXTOP(Arg,ld),x),o));
}
/* emits a try, retry or trust, optimizing for a particular case */
static void
emit_cp_inst(compiler_vm_op op, yamop * Address, int Flag, int NClausesAfter)
{
int HasCut = clause_has_cut(Address);
indexed_code_for_cut = NIL;
if (op != try_op && profiling)
emit(retry_profiled_op, Unsigned(CurrentPred), Zero);
else if (op != try_op && call_counting)
emit(count_retry_op, Unsigned(CurrentPred), Zero);
if (NGroups == 1)
Flag = Flag | LoneGroup;
else if (Flag & LastGroup) {
if (op == try_op) {
Flag &= (FirstIndex | LastFoundList | HeadIndex | IsAtom);
switch (Flag) {
case 0:
emit_tr(trylf_op, Body(Address), NClausesAfter, HasCut);
return;
case FirstIndex | LastFoundList:
emit_tr(trylf_op, SecLB(Address), NClausesAfter, HasCut);
return;
case FirstIndex | IsAtom:
emit_tr(retry_op, SecB(Address), NClausesAfter, HasCut);
return;
case FirstIndex:
emit_tr(trylf_op, SecB(Address), NClausesAfter, HasCut);
return;
case HeadIndex | LastFoundList:
emit_tr(trylh_op, ThiLB(Address), NClausesAfter, HasCut);
return;
case HeadIndex | IsAtom:
emit_tr(trylf_op, ThiB(Address), NClausesAfter, HasCut);
return;
case HeadIndex:
emit_tr(trylh_op, ThiB(Address), NClausesAfter, HasCut);
return;
}
} else
Flag ^= LastGroup;
Flag |= LoneGroup;
}
if (ExtendedSingle) {
if (op == trust_op) {
emit_cp_inst(retry_op, Address, Flag, NClausesAfter);
emit_tr(trust_op, FailAddress, NClausesAfter-1, HasCut);
return;
}
Flag |= LoneGroup;
}
switch (Flag) {
case 0:
emit_try(op, tryin_op - try_op, Body(Address), NClausesAfter, HasCut);
return;
case LoneGroup:
emit_try(op, try_op - try_op, Body(Address), NClausesAfter, HasCut);
return;
case FirstIndex:
emit_try(op, tryfin_op - try_op, SecB(Address), NClausesAfter, HasCut);
return;
case FirstIndex | IsAtom:
emit_try(op, tryin_op - try_op, SecB(Address), NClausesAfter, HasCut);
return;
case FirstIndex | LastFoundList:
emit_try(op, tryfin_op - try_op, SecLB(Address), NClausesAfter, HasCut);
return;
case FirstIndex | LoneGroup:
emit_try(op, tryf_op - try_op, SecB(Address), NClausesAfter, HasCut);
return;
case FirstIndex | LoneGroup | IsAtom:
emit_try(op, try_op - try_op, SecB(Address), NClausesAfter, HasCut);
return;
case FirstIndex | LoneGroup | LastFoundList:
emit_try(op, tryf_op - try_op, SecLB(Address), NClausesAfter, HasCut);
return;
case HeadIndex:
emit_try(op, tryhin_op - try_op, ThiB(Address), NClausesAfter, HasCut);
return;
case HeadIndex | LastFoundList:
emit_try(op, tryhin_op - try_op, ThiLB(Address), NClausesAfter, HasCut);
return;
case HeadIndex | IsAtom:
emit_try(op, trytin_op - try_op, ThiB(Address), NClausesAfter, HasCut);
return;
case HeadIndex | LoneGroup:
emit_try(op, tryh_op - try_op, ThiB(Address), NClausesAfter, HasCut);
return;
case HeadIndex | LoneGroup | LastFoundList:
emit_try(op, tryh_op - try_op, ThiLB(Address), NClausesAfter, HasCut);
return;
case HeadIndex | LoneGroup | IsAtom:
emit_try(op, tryt_op - try_op, ThiB(Address), NClausesAfter, HasCut);
return;
default:
return;
}
}
/* emits a large switch instruction */
static CELL
emit_space(compiler_vm_op op, int space, int nof)
{
labelno += 2;
emit(label_op, labelno, Zero);
StorePoint = emit_extra_size(op, Unsigned(nof), space);
return (labelno);
}
/* emits a go instruction */
static CELL
emit_go(int Gender, Term Name)
{
labelno += 2;
emit(label_op, labelno, Zero);
if (Gender == ApplCl)
StorePoint = emit_extra_size(go_f_op, Zero, 3 * CellSize);
else
StorePoint = emit_extra_size(go_c_op, Zero, 3 * CellSize);
*StorePoint++ = Unsigned(Name);
StorePoint[1] = (CELL)FailAddress;
return (labelno);
}
/* emits an if_not_then instruction */
static void
emit_if_not(Term T1, CELL Ad1, CELL Ad2)
{
StorePoint = emit_extra_size(if_not_op, Zero, 3 * CellSize);
*StorePoint++ = Unsigned(T1);
*StorePoint++ = Unsigned(Ad1);
StorePoint[0] = Unsigned(Ad2);
}
/* places the code where to go in a go instruction */
static void
fill_go(CELL *LCons, CELL Code)
{
LCons[0] = Code;
}
/* fills a space after a first instruction with four addresses */
static void
fill_switch_slots(CELL *WhereTo, CELL FAddr, CELL SecAddr, CELL ThirdAddr, CELL ThourthAddr, int NOf)
{
*WhereTo++ = FAddr;
*WhereTo++ = SecAddr;
*WhereTo++ = ThirdAddr;
if (NOf > 3)
*WhereTo = ThourthAddr;
}
/* Places in a space pairs 0 -> Fail */
static void
ClrSpace(CELL *StSpace, int NBytes)
{
int i;
NBytes /= 2 * CellSize;
for (i = 0; i < NBytes; i++) {
*StSpace++ = Zero;
*StSpace++ = (CELL)FailAddress;
}
}
/* Evaluates the number of groups and builds the clause and group arrays */
static int
NGroupsIn(PredEntry *ap)
{
int x, y, PresentGroup;
ClauseDef *ActualCl = ArOfCl, *LastClauses[MaxOptions];
GroupDef *Group = Groups;
yamop *q = (yamop *)(ap->FirstClause), *LastOne = (yamop *)(ap->LastClause);
NGroups = 1;
LastClauses[VarCl] = NIL;
LastClauses[AtCl] = NIL;
LastClauses[ApplCl] = NIL;
LastClauses[ListCl] = NIL;
Group = Groups;
NClauses = 0;
while (IsVarClause(q)) {
RemovedCl = TRUE;
if (q == LastOne)
return (NGroups);
else
q = (yamop *)NextClause(q);
}
x = KindOfArg(q);
if (NonVarCl(x))
PresentGroup = 1;
else
PresentGroup = 0;
LastClauses[x] = Group->Start = ActualCl;
Group->NCl = 0;
Group->First = q;
Group->Type[VarCl] = 0;
Group->Type[AtCl] = 0;
Group->Type[ApplCl] = 0;
Group->Type[ListCl] = 0;
Group->SInfo = OnlyNils;
do {
while (IsVarClause(q)) {
RemovedCl = TRUE;
if (q == LastOne)
return (NGroups);
else
q = (yamop *)NextClause(q);
}
x = KindOfArg(q);
NClauses++;
ActualCl->Kind = x;
ActualCl->Code = q;
if (x == ListCl) {
ActualCl->Name = HeadOfList(q);
y = 1;
} else if (NonVarCl(x)) {
ActualCl->Name = TermOfCl(q);
y = 1;
} else {
y = 0;
}
if (y != PresentGroup) {
Group++->Last = (ActualCl - 1)->Code;
NGroups++;
if ((ADDR)Group > TrailTop-1024)
growtrail(64 * 1024L);
Group->First = q;
Group->Start = ActualCl;
Group->NCl = 0;
Group->Type[VarCl] = 0;
Group->Type[AtCl] = 0;
Group->Type[ApplCl] = 0;
Group->Type[ListCl] = 0;
Group->SInfo = OnlyNils;
LastClauses[VarCl] = NIL;
LastClauses[AtCl] = NIL;
LastClauses[ApplCl] = NIL;
LastClauses[ListCl] = NIL;
PresentGroup = y;
}
if (x == AtCl) {
if (ActualCl->Name != MkAtomTerm(AtomNil))
Group->SInfo &= ~OnlyNils;
if (KindOfBipArg(q) & (FIsAtom|FIsNum|FIsPrimi)) {
Group->SInfo |= UsesBips;
ActualCl->Name = 0x0;
}
}
ActualCl->Next = NIL;
if (LastClauses[x] != NIL)
LastClauses[x]->Next = ActualCl;
LastClauses[x] = ActualCl++;
/* check for overflow in case we have a really big database */
if (ASP <= CellPtr (ActualCl) + 256) {
freep = (char *)ActualCl;
save_machine_regs();
longjmp(CompilerBotch, 3);
}
(Group->Type[x])++;
(Group->NCl)++;
if (q == LastOne)
q = NIL;
else
q = (yamop *)NextClause(q);
} while (q != NIL);
return (NGroups);
}
/* for each group find out how many clauses follow that group */
static void
CountGroups(GroupDef *Gr, int NGr)
{
GroupDef *grp = Gr+NGr;
int cls = 0;
do {
grp--;
grp->NofClausesAfter = cls;
cls += grp->NCl;
} while(grp > Gr);
}
/* Finds the first clause whose arg is neither a list or a var */
static yamop *
FindFirst(int i, int kind)
{
ClauseDef *GrClaus = Groups[i].Start;
while (GrClaus->Kind != kind)
GrClaus++;
if (AtomsOnlyNil)
return (SecB(GrClaus->Code));
return (Body(GrClaus->Code));
}
/* evaluates the size (power of two) necessary for a table */
static int
SizeTable(int Cases)
{
register int i = 2;
if (Cases <= MinHashEntries)
return (Cases);
while ((Cases /= 2) > 0)
i *= 2;
return (i * 2);
}
#define HASH_SHIFT 6
/*
* creates the hash table works if TableSize is a power of two
*/
static void
BuildHash(CELL *WhereTo, int NOfEntries, int TableSize, int Gend)
{
register int hash;
register int i;
#ifdef DEBUG
#ifdef CLASHES
int clashes = 0;
#endif /* CLASHES */
#endif /* DEBUG */
Term WorkTerm;
EntryDef *EntryP = Entries;
yamop *Base = (yamop *)CodePtr(WhereTo);
yamop *EndSpace = (yamop *)((char *)Base + TableSize * 2 * CellSize);
ClrSpace(WhereTo, ((int) ((char *)EndSpace - (char *)Base)));
TableSize--;
for (i = 0; i < NOfEntries; ++i) {
register Int d;
CELL *hentry;
WorkTerm = EntryP->Class;
hash = (Unsigned(WorkTerm) >> HASH_SHIFT) & TableSize;
hentry = (CELL *)Base + hash * 2;
d = TableSize & (Unsigned(WorkTerm) | 1);
while (*hentry) {
#ifdef DEBUG
#ifdef CLASHES
++clashes;
#endif /* CLASHES */
#endif /* DEBUG */
hash = (hash + d) & TableSize;
hentry = (CELL *)Base + hash * 2;
}
hentry[0] = Unsigned(WorkTerm);
hentry[1] = (CELL)((EntryP++)->Code);
}
#ifdef DEBUG
#ifdef CLASHES
YP_fprintf(YP_stderr,"hash table clashes: %d %d\n", clashes, NOfEntries);
#endif /* CLASHES */
#endif /* DEBUG */
}
static void
BuildIfTable(CELL *WhereTo, int NOfEntries)
{
register EntryDef *EntryP = Entries;
register int j;
*WhereTo++ = (CELL)FailAddress;
for (j = 0; j < NOfEntries; ++j) {
*WhereTo++ = Unsigned(EntryP->Class);
*WhereTo++ = (CELL)((EntryP++)->Code);
}
}
/* Creates the code for each entry in a group */
static void
TreatEntry(EntryDef *Entrance, int Gender, int PositionFlag, GroupDef *Gr)
{
if (Entrance->Last == Entrance->First) {
if (PositionFlag & LastGroup) {
/* last group, meaning we already have a choice point set */
register yamop * k = (Entrance->First)->Code;
labelno += 2;
emit(label_op, Entrance->Code = labelno, Zero);
if (PositionFlag & HeadIndex) {
emit_tr(trust_op, ThiB(k), 1, clause_has_cut(k));
} else {
/* we cannot emit to SecLB because switch might have already
set SREG :-(, hence making us jump one step ahead. This
is not a problem with a direct jump, in this case the
work on S will just have been squandered */
emit_tr(trust_op, Body(k), 1, clause_has_cut(k));
}
} else if (ExtendedSingle) {
/* a single alternative and a catchall clause */
register yamop * k = (Entrance->First)->Code;
labelno += 2;
emit(label_op, Entrance->Code = labelno, Zero);
/* if we are in a list */
if (PositionFlag & HeadIndex) {
/* we cannot emit to SecLB because switch might have already
set SREG :-( */
emit_try(try_op, 0, Body(k), 1, clause_has_cut(k));
if (indexed_code_for_cut != NIL) {
Entrance->Code =
(CELL)indexed_code_for_cut;
} else {
emit_tr(trust_op, FailAddress, 1, clause_has_cut(k));
Entrance->Code = labelno;
}
} else {
emit_try(try_op, 0, SecB(k), 2, clause_has_cut(k));
if (indexed_code_for_cut != NIL) {
Entrance->Code =
(CELL)indexed_code_for_cut;
} else {
emit_tr(trust_op, FailAddress, 1, clause_has_cut(k));
Entrance->Code = labelno;
}
}
} else if (PositionFlag & HeadIndex) {
/* single clause in the middle of code, no need to worry about
choicepoints */
Entrance->Code = Unsigned(ThiB((Entrance->First)->Code));
} else {
Entrance->Code = Unsigned(SecB((Entrance->First)->Code));
}
} else {
ClauseDef *k = Entrance->First;
int nofentries = 1, nofalts;
while (k != Entrance->Last) {
nofentries++;
k = k->Next;
}
nofalts = Gr->NofClausesAfter+nofentries;
k = Entrance->First;
labelno += 2;
emit(label_op, (Entrance->Code = labelno), Zero);
emit_cp_inst(try_op, k->Code, PositionFlag, nofalts);
nofalts--;
if (indexed_code_for_cut != NIL) {
Entrance->Code = (CELL)indexed_code_for_cut;
} else {
while ((k = k->Next) != Entrance->Last) {
emit_cp_inst(retry_op, k->Code, PositionFlag, nofalts);
nofalts--;
}
emit_cp_inst(trust_op, k->Code, PositionFlag, nofalts);
/* emit a jump with the place to jump to after finishing this group */
if (NGroups > 1 && !(PositionFlag & LastGroup) && !ExtendedSingle)
emit(jump_op, (CELL)((Gr+1)->First), Zero);
}
}
}
/* Creates the code for either appl or atomic cases */
static CELL
DealFixed(ClauseDef *j, int Gender, compiler_vm_op op, int Flag, GroupDef *Gr)
{
int NDiffTerms = 1;
int TableSize, k;
ClauseDef *NextInChain;
Term HeadName;
CELL LFixed;
CELL *WhereToStore;
EntryDef *Entry = Entries;
Entry->Class = j->Name;
Entry->Last = Entry->First = j;
do {
int l = 0;
HeadName = j->Name;
NextInChain = j->Next;
j->Next = NIL;
Entry = Entries;
while (l < NDiffTerms && Entry->Class != HeadName)
Entry++, l++;
if (l == NDiffTerms) {
if ((ADDR)Entry > TrailTop-1024)
growtrail(64 * 1024L);
Entry->Class = HeadName;
Entry->Last = Entry->First = j;
NDiffTerms++;
} else {
(Entry->Last)->Next = j;
Entry->Last = j;
}
} while ((j = NextInChain) != NIL && j->Name != 0x0);
Entry = Entries;
if (NDiffTerms == 1) {
CELL *WhereToStore;
if (AtomsOnlyNil) {
TreatEntry(Entry, Gender, Flag, Gr);
return ((CELL)(Entry->Code));
}
LFixed = emit_go(Gender, Entry->Class);
WhereToStore = StorePoint;
TreatEntry(Entry, Gender, Flag, Gr);
fill_go(WhereToStore, (CELL)(Entry->Code));
return (LFixed);
}
TableSize = SizeTable(NDiffTerms);
if (NDiffTerms <= MinHashEntries) {
if (op == switch_c_op)
op = if_c_op;
else
op = if_f_op;
LFixed = emit_space(op, CellSize + TableSize * 2 * CellSize, TableSize);
} else
LFixed = emit_space(op, TableSize * 2 * CellSize, TableSize);
WhereToStore = StorePoint;
for (k = 0; k < NDiffTerms; k++) {
TreatEntry(Entry, Gender, Flag, Gr);
Entry++;
}
if (NDiffTerms > MinHashEntries)
BuildHash(WhereToStore, NDiffTerms, TableSize, Gender);
else
BuildIfTable(WhereToStore, NDiffTerms);
return (LFixed);
}
/*
If atom(X) and friends are around, and for the moment, just forget
about doing indexing for constants and generate a traditional try,
retry chain.
*/
static CELL
DealFixedWithBips(ClauseDef *j, int NClauses, int Flag, GroupDef *Gr)
{
int i = 2;
CELL my_labelno;
int nofalts = Gr->NofClausesAfter+NClauses;
labelno += 2;
my_labelno = labelno;
emit(label_op, labelno, Zero);
#ifdef AGRESSIVE_BIPS
if (j->Name != 0x0 && j->Next->Name != 0x0) {
/*
we've got a sequence of i clauses with known argument. Can
index as normal for them
*/
CELL old_labelno;
yamop *old_FailAddress;
int old_ExtendedSingle, old_NAlts;
ClauseDef *j0 = j;
labelno += 2;
old_labelno = labelno;
old_FailAddress = FailAddress;
FailAddress = labelno;
/* the clauses behaves as if a new group */
NGroups++;
old_ExtendedSingle = ExtendedSingle;
ExtendedSingle = FALSE;
j = j->Next;
old_NAlts = Gr->NofClausesAfter;
while (j->Name != 0x0) i++, j = j->Next;
Gr->NofClausesAfter = old_NAlts + G->NCl - i;
DealFixed(j0, AtCl, switch_c_op, FirstIndex | IsAtom, Gr);
emit(label_op, old_labelno, Zero);
FailAddress = old_FailAddress;
ExtendedSingle = old_ExtendedSingle;
Gr->NofClausesAfter = old_NAlts;
NGroups--;
} else {
#endif /* AGRESSIVE_BIPS */
emit_cp_inst(try_op, j->Code, Flag, nofalts);
if (indexed_code_for_cut != NIL) {
/* Unfortunately, this is bound never to happen */
return((CELL)indexed_code_for_cut);
}
else j = j->Next;
nofalts--;
#ifdef AGRESSIVE_BIPS
}
#endif /* AGRESSIVE_BIPS */
/* we handled a group */
for (; i < NClauses; i++) {
emit_cp_inst(retry_op, j->Code, Flag, nofalts);
j = j->Next;
nofalts--;
}
emit_cp_inst(trust_op, j->Code, Flag, nofalts);
/* emit a jump with the place to jump to after finishing this group */
if (NGroups > 1 && !(Flag & LastGroup) && !ExtendedSingle)
emit(jump_op, (CELL)((Gr+1)->First), Zero);
return(my_labelno);
}
static CELL
DealCons(int i)
{
int NAtCl = Groups[i].Type[AtCl];
ClauseDef *Cla = Groups[i].Start;
if (NAtCl == 1) {
if (FinalGr(i)) {
yamop * Cl = FindFirst(i, AtCl);
labelno += 2;
emit(label_op, labelno, Zero);
emit_tr(trust_op, Cl, 1, clause_has_cut(Cl));
return (labelno);
} else if (ExtendedSingle) {
yamop * Cl = FindFirst(i, AtCl);
labelno += 2;
emit(label_op, labelno, Zero);
emit_tr(try_op, Cl, 2, clause_has_cut(Cl));
emit_tr(trust_op, FailAddress, 1, clause_has_cut(FailAddress));
return (labelno);
}
return ((CELL)FindFirst(i, AtCl));
} else if (NAtCl == 0)
return ((CELL)FailAddress);
while (Cla->Kind != AtCl)
Cla++;
if (FinalGr(i)) {
if (Groups[i].SInfo & UsesBips)
return(DealFixedWithBips(Cla, NAtCl, LastGroup, Groups+i));
else
return (DealFixed(Cla, AtCl, switch_c_op,
FirstIndex | IsAtom | LastGroup, Groups+i));
} else {
if (Groups[i].SInfo & UsesBips)
return(DealFixedWithBips(Cla, NAtCl, 0, Groups+i));
else
return (DealFixed(Cla, AtCl, switch_c_op,
FirstIndex | IsAtom, Groups+i));
}
}
static CELL
DealAppl(int i)
{
int NApCl = Groups[i].Type[ApplCl];
ClauseDef *Cla = Groups[i].Start;
if (NApCl == 1) {
if (FinalGr(i)) {
yamop * Cl = FindFirst(i, ApplCl);
labelno += 2;
emit(label_op, labelno, Zero);
emit_tr(trust_op, Cl, 1, clause_has_cut(Cl));
return (labelno);
} else if (ExtendedSingle) {
yamop * Cl = FindFirst(i, ApplCl);
labelno += 2;
emit(label_op, labelno, Zero);
emit_tr(try_op, Cl, 2, clause_has_cut(Cl));
emit_tr(trust_op, FailAddress, 1, clause_has_cut(FailAddress));
return (labelno);
}
return ((CELL)FindFirst(i, ApplCl));
} else if (NApCl == 0)
return ((CELL)FailAddress);
while (Cla->Kind != ApplCl)
Cla++;
if (FinalGr(i)) {
int Flag = FirstIndex | LastGroup;
return (DealFixed(Cla, ApplCl, switch_f_op, Flag, Groups+i));
}
return (DealFixed(Cla, ApplCl, switch_f_op, FirstIndex, Groups+i));
}
/* Finds the body of the first clause of a group whose first arg is a list */
static CELL
StartList(int i)
{
ClauseDef *j = Groups[i].Start;
while (j->Kind != ListCl)
j++;
if (FinalGr(i)) {
labelno += 2;
emit(label_op, labelno, Zero);
emit_tr(trust_op, SecLB(j->Code), 1, clause_has_cut(j->Code));
return (labelno);
} else if (ExtendedSingle) {
labelno += 2;
emit(label_op, labelno, Zero);
emit_tr(try_op, SecLB(j->Code), 2, clause_has_cut(j->Code));
emit_tr(trust_op, FailAddress, 1, clause_has_cut(FailAddress));
return (labelno);
} else {
return (Unsigned(SecLB(j->Code)));
}
}
static CELL
DealLAt(ClauseDef *Cla, int NOfClau, int NG)
{
if (NOfClau == 0)
return ((CELL)FailAddress);
if (FinalGr(NG)) {
int Flag = HeadIndex | IsAtom | LastGroup;
return (DealFixed(Cla, AtCl, switch_c_op, Flag, Groups+NG));
}
return (DealFixed(Cla, AtCl, switch_c_op, HeadIndex | IsAtom, Groups+NG));
}
static CELL
DealLAppl(ClauseDef *Cla, int NOfClauses, int NG)
{
if (NOfClauses == 0)
return ((CELL)FailAddress);
if (FinalGr(NG)) {
int Flag = HeadIndex | LastGroup;
return (DealFixed(Cla, ApplCl, switch_f_op, Flag, Groups+NG));
}
return (DealFixed(Cla, ApplCl, switch_f_op, HeadIndex, Groups+NG));
}
static CELL
DealLList(ClauseDef *j, int NOfClauses, int PositionFlag, int NG)
{
int k, nofalts = 1;
if (NOfClauses == 0)
return ((CELL)FailAddress);
else if (NOfClauses == 1) {
if (FinalGr(NG)) {
labelno += 2;
emit(label_op, labelno, Zero);
if (PositionFlag & FirstIndex)
emit_tr(trust_op, SecLB(j->Code), 1, clause_has_cut(j->Code));
else
emit_tr(trust_op, ThiLB(j->Code), 1, clause_has_cut(j->Code));
return (labelno);
} else if (ExtendedSingle) {
labelno += 2;
emit(label_op, labelno, Zero);
if (PositionFlag & FirstIndex)
emit_tr(try_op, SecLB(j->Code), 2, clause_has_cut(j->Code));
else
emit_tr(try_op, ThiLB(j->Code), 2, clause_has_cut(j->Code));
emit_tr(trust_op, FailAddress, 1, clause_has_cut(FailAddress));
return (labelno);
}
if (PositionFlag & FirstIndex)
return (Unsigned(SecLB(j->Code)));
return (Unsigned(ThiLB(j->Code)));
}
if (FinalGr(NG))
PositionFlag |= LastGroup;
labelno += 2;
emit(label_op, labelno, Zero);
nofalts = Groups[NG].NofClausesAfter+NOfClauses;
emit_cp_inst(try_op, j->Code, PositionFlag, nofalts);
nofalts--;
if (indexed_code_for_cut != NIL) {
return((CELL)indexed_code_for_cut);
} else {
j = j->Next;
for (k = 2; k < NOfClauses; ++k) {
emit_cp_inst(retry_op, j->Code, PositionFlag, nofalts);
nofalts--;
j = j->Next;
}
emit_cp_inst(trust_op, j->Code, PositionFlag, nofalts);
/* emit a jump with the place to jump to after finishing this group */
if (NGroups > 1 && !(PositionFlag & LastGroup) && !ExtendedSingle)
emit(jump_op, (CELL)(Groups[NG+1].First), Zero);
return (labelno);
}
}
static int
NoHeadVar(ClauseDef *CInfo, int NClauses)
{
while (NClauses--) {
if (KindOfListArg(CInfo->Code) == VarCl)
return (FALSE);
CInfo = CInfo->Next;
}
return (TRUE);
}
/* Creates code when several clauses first arg is a list */
static CELL
DealList(int i)
{
int k;
ClauseDef *j;
CELL LLFirst;
CELL *WhereToStore;
int NListCl = Groups[i].Type[ListCl];
int VFlags;
int nofalts = Groups[i].NofClausesAfter+NListCl;
if (NListCl == 0)
return ((CELL)FailAddress);
if (NListCl == 1)
return (StartList(i));
j = Groups[i].Start;
while (j->Kind != ListCl)
j++;
if (NoHeadVar(j, NListCl)) {
CELL LLVar, LLAt, LLList, LLAppl;
ClauseDef *Firsts[MaxOptions], *Lasts[MaxOptions];
int NOfSameCl[MaxOptions], l;
NOfSameCl[AtCl] = 0;
NOfSameCl[VarCl] = 0;
NOfSameCl[ListCl] = 0;
NOfSameCl[ApplCl] = 0;
LLFirst = emit_space(switch_h_op, 4 * CellSize, 0);
WhereToStore = StorePoint;
LLVar = DealLList(j, NListCl, LastFoundList | FirstIndex, i);
for (l = 0; l < NListCl; ++l) {
int z = KindOfListArg(j->Code);
if (NOfSameCl[z] != 0)
Lasts[z]->Next = j;
else
Firsts[z] = j;
Lasts[z] = j;
(NOfSameCl[z])++;
j = j->Next;
}
if (NOfSameCl[AtCl])
Lasts[AtCl]->Next = NIL;
if (NOfSameCl[VarCl])
Lasts[VarCl]->Next = NIL;
if (NOfSameCl[ListCl])
Lasts[ListCl]->Next = NIL;
if (NOfSameCl[ApplCl])
Lasts[ApplCl]->Next = NIL;
LLAppl = DealLAppl(Firsts[ApplCl], NOfSameCl[ApplCl], i);
LLAt = DealLAt(Firsts[AtCl], NOfSameCl[AtCl], i);
LLList = DealLList(Firsts[ListCl], NOfSameCl[ListCl],
LastFoundList | HeadIndex, i);
fill_switch_slots(WhereToStore, LLList, LLAt, LLAppl, LLVar, 4);
return (LLFirst);
}
if (FinalGr(i))
VFlags = FirstIndex | LastFoundList | LastGroup;
else
VFlags = FirstIndex | LastFoundList;
labelno += 2;
emit(label_op, labelno, Zero);
emit_cp_inst(try_op, j->Code, VFlags, nofalts);
nofalts--;
if (indexed_code_for_cut != NIL) {
return((CELL)indexed_code_for_cut);
} else {
j = j->Next;
for (k = 2; k < NListCl; ++k) {
emit_cp_inst(retry_op, j->Code, VFlags, nofalts);
j = j->Next;
nofalts--;
}
emit_cp_inst(trust_op, j->Code, VFlags, nofalts);
/* emit a jump with the place to jump to after finishing this group */
if (NGroups > 1 && !(VFlags & LastGroup) && !ExtendedSingle)
emit(jump_op, (CELL)(Groups[i+1].First), Zero);
return (labelno);
}
}
/* Finds the place where to go if we have a flop */
static CELL
GetFailToGo(int NG)
{
if (NGroups == 1)
return (Unsigned(FAILCODE));
if (FinalGr(NG))
return (Unsigned(TRUSTFAILCODE));
if (ExtendedSingle)
return (Unsigned(Body(Groups[NG + 1].Start->Code)));
return (Unsigned(FAILCODE));
}
/* Verifies if we have a group and a var first clause (catchall, usually) */
static int
IsExtendedSingle(int NG)
{
if (NGroups == 2 && NG == 0 && Groups[1].NCl == 1 &&
NClauses <= CLAUSES_FOR_EXTENDED_SINGLE)
return (TRUE);
return (FALSE);
}
static int
gr_has_cuts(GroupDef *gr)
{
return(0);
}
static void
EmitGrSwitch(int Count)
{
GroupDef Gr;
Gr = Groups[Count];
if (!Gr.Type[ApplCl] && (Gr.SInfo & OnlyNils)) {
AtomsOnlyNil = TRUE;
if (FinalGr(Count)) {
emit_space(switch_ll_op, 3 * CellSize,
((Gr.NCl) << 1)+gr_has_cuts(Groups+Count));
} else if (NGroups == 1 || ExtendedSingle)
emit_space(switch_lnl_op, 4 * CellSize, 0);
else
emit_space(switch_nvl_op, 3 * CellSize, 0);
} else {
AtomsOnlyNil = FALSE;
if (FinalGr(Count))
emit_space(switch_l_op, 3 * CellSize,
((Gr.NCl) << 1)+gr_has_cuts(Groups+Count));
else if (NGroups == 1 || ExtendedSingle)
emit_space(switch_t_op, 4 * CellSize, 0);
else
emit_space(switch_nv_op, 3 * CellSize, 0);
}
}
/* Creates the indexation code for a non var group */
static int
IndexNonVarGr(int Count)
{
CELL LVar, LCons, LA, LL;
GroupDef Gr;
CELL *WhereToStore;
Gr = Groups[Count];
ExtendedSingle = IsExtendedSingle(Count);
FailAddress = (yamop *)GetFailToGo(Count);
if (NGroups > 1 && !(ExtendedSingle) && Count < NGroups - 1) {
labelno += 2;
Groups[Count + 1].First = (yamop *)labelno;
}
if (Gr.NCl == 1) {
yamop * Cl = (Gr.Start)->Code;
if (Count == 0)
emit_tr(try_op, Body(Cl), Gr.NofClausesAfter+1, clause_has_cut(Cl));
else if (FinalGr(Count))
emit_tr(trust_op, Body(Cl), Gr.NofClausesAfter+1, clause_has_cut(Cl));
else
emit_tr(retry_op, Body(Cl), Gr.NofClausesAfter+1, clause_has_cut(Cl));
return (FALSE); /* Indexation is not necessary */
}
if (NGroups > 1 && !(ExtendedSingle) && Count < NGroups - 1) {
if (Count == 0)
emit_tr(tryme_op, (yamop *) labelno,
Gr.NofClausesAfter+1, clause_has_cut((Gr.Start)->Code));
else
emit_tr(retryme_op, (yamop *) labelno,
Gr.NofClausesAfter+1, clause_has_cut((Gr.Start)->Code));
}
EmitGrSwitch(Count);
WhereToStore = StorePoint;
LCons = DealCons(Count);
AtomsOnlyNil = FALSE;
LA = DealAppl(Count);
LL = DealList(Count);
if (log_update) {
LVar = log_update;
} else
LVar = Unsigned(FirstCl);
if (NGroups == 1 || ExtendedSingle)
fill_switch_slots(WhereToStore, LL, LCons, LA, LVar, 4);
else
fill_switch_slots(WhereToStore, LL, LCons, LA, LVar, 3);
return (TRUE);
}
static void
IndexVarGr(int Count)
{
GroupDef *Gr;
ClauseDef *Cla;
int j;
int nofalts;
Gr = Groups + Count;
Cla = Gr->Start;
nofalts = Gr->NofClausesAfter+Gr->NCl;
if (Count == 0) {
emit_tr(try_op, Body((Cla)->Code), nofalts, clause_has_cut(Cla->Code));
Cla++;
nofalts--;
} else if (Count == NGroups - 1 && Gr->NCl == 1) {
emit(label_op, Unsigned(Gr->First), Zero);
emit_tr(trust_op, Body((Cla)->Code), 1, clause_has_cut(Cla->Code));
return;
} else {
emit(label_op, Unsigned(Gr->First), Zero);
emit_tr(retry_op, Body((Cla)->Code), nofalts, clause_has_cut(Cla->Code));
Cla++;
nofalts--;
}
for (j = 2; j < Gr->NCl; ++j) {
emit_tr(retry_op, Body((Cla)->Code), nofalts, clause_has_cut(Cla->Code));
Cla++;
nofalts--;
}
if (Gr->NCl > 1) {
if (Count == NGroups - 1)
emit_tr(trust_op, Body(Cla->Code), 1, clause_has_cut(Cla->Code));
else
emit_tr(retry_op, Body(Cla->Code), nofalts, clause_has_cut(Cla->Code));
}
}
static int
SimpleCase(void)
{
if (Groups[0].Type[VarCl] != 0)
return (FALSE);
return (IndexNonVarGr(0));
}
static int
ComplexCase(void)
{
int Indexable = FALSE, i;
if (IsExtendedSingle(0))
return (SimpleCase());
emit(jump_v_op, (CELL) FirstCl, Zero);
if (Groups[0].Type[VarCl] == 0)
i = 0;
else {
IndexVarGr(0);
i = 1;
}
for (; i < NGroups; i += 2) {
Indexable |= IndexNonVarGr(i);
if (i < NGroups - 1)
IndexVarGr(i + 1);
}
return (Indexable);
}
static int
SpecialCases(void)
{
CELL LVar;
if (log_update) {
LVar = log_update;
} else {
LVar = (CELL)(ArOfCl->Code);
}
/* Clear what was left before */
freep = (char *) (ArOfCl + NClauses);
CodeStart = cpc = NIL;
/* For now just a special case */
if (NGroups == 2 && NClauses == 2 && Groups[0].Type[AtCl] == 1 &&
!RemovedCl && !(Groups[0].SInfo & UsesBips)) {
emit_if_not(ArOfCl->Name, Unsigned(Body(ArOfCl[1].Code)),
LVar);
return (TRUE);
} else
return (FALSE);
}
CODEADDR
PredIsIndexable(PredEntry *ap)
{
int NGr, Indexable = 0;
CODEADDR indx_out = NIL;
log_update = 0;
if (setjmp(CompilerBotch) == 3) {
/* just duplicate the stack */
restore_machine_regs();
gc(ap->ArityOfPE, ENV, CP);
}
restart_index:
ErrorMessage = NULL;
labelno = 1;
RemovedCl = FALSE;
FirstCl = (yamop *)(ap->FirstClause);
CurrentPred = ap;
if (CurrentPred->PredFlags & ProfiledPredFlag)
profiling = TRUE;
else if (CurrentPred->PredFlags & CountPredFlag)
call_counting = TRUE;
else
profiling = FALSE;
IPredArity = ap->ArityOfPE;
/* Store the clauses in the Global */
ArOfCl = (ClauseDef *) H;
/* and the groups in the Auxiliary */
Groups = (GroupDef *) TR;
NGr = NGroupsIn(ap);
CountGroups(Groups, NGr);
/* store entries after groups */
Entries = (EntryDef *) (Groups + NGroups);
CodeStart = cpc = NIL;
freep = (char *) (ArOfCl + NClauses);
if (ErrorMessage != NULL) {
return (NIL);
}
if (CurrentPred->PredFlags & LogUpdatePredFlag) {
log_update = labelno;
labelno += 2;
}
if (NClauses == 0) {
Indexable = FALSE;
return(NIL);
} else {
if (NGr == 1)
Indexable = SimpleCase();
else
Indexable = ComplexCase();
if (!Indexable)
Indexable = SpecialCases();
}
if (CellPtr(freep) >= ASP) {
Error(SYSTEM_ERROR, TermNil, "out of stack space while indexing");
return(NIL);
}
if (log_update && NClauses > 1) {
int i;
Clause *cl;
Indexable = TRUE;
emit(label_op, log_update, Zero);
emit(try_op, Unsigned(Body(ArOfCl[0].Code)), Zero);
cl = ClauseCodeToClause(ArOfCl[0].Code);
if (cl->ClFlags & LogUpdRuleMask) {
cl->u2.ClExt->u.EC.ClRefs++;
} else {
cl->u2.ClUse++;
}
for (i = 1; i < NClauses-1; i++) {
emit(retry_op, Unsigned(Body(ArOfCl[i].Code)), Zero);
cl = ClauseCodeToClause(ArOfCl[0].Code);
if (cl->ClFlags & LogUpdRuleMask) {
cl->u2.ClExt->u.EC.ClRefs++;
} else {
cl->u2.ClUse++;
}
}
emit(trust_op, Unsigned(Body(ArOfCl[i].Code)), Zero);
cl = ClauseCodeToClause(ArOfCl[i].Code);
if (cl->ClFlags & LogUpdRuleMask) {
cl->u2.ClExt->u.EC.ClRefs++;
} else {
cl->u2.ClUse++;
}
}
if (!Indexable) {
return (NIL);
} else {
#ifdef DEBUG
if (Option['i' - 'a' + 1]) {
ShowCode();
}
#endif
if ((indx_out = assemble(ASSEMBLING_INDEX)) == NIL) {
if (!growheap(FALSE)) {
Error(SYSTEM_ERROR, TermNil, ErrorMessage);
return(FALSE);
}
goto restart_index;
}
}
return(indx_out);
}