865 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			865 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
| /*************************************************************************
 | |
| *									 *
 | |
| *	 YAP Prolog 							 *
 | |
| *									 *
 | |
| *	Yap Prolog was developed at NCCUP - Universidade do Porto	 *
 | |
| *									 *
 | |
| * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 *
 | |
| *									 *
 | |
| **************************************************************************
 | |
| *									 *
 | |
| * File:		computils.c						 *
 | |
| * comments:	some useful routines for YAP's compiler			 *
 | |
| *									 *
 | |
| * Last rev:     $Date: 2007-11-26 23:43:08 $							 *
 | |
| * $Log: not supported by cvs2svn $
 | |
| * Revision 1.31  2007/11/06 17:02:12  vsc
 | |
| * compile ground terms away.
 | |
| *
 | |
| * Revision 1.30  2006/09/20 20:03:51  vsc
 | |
| * improve indexing on floats
 | |
| * fix sending large lists to DB
 | |
| *
 | |
| * Revision 1.29  2005/12/05 17:16:10  vsc
 | |
| * write_depth/3
 | |
| * overflow handlings and garbage collection
 | |
| * Several ipdates to CLPBN
 | |
| * dif/2 could be broken in the presence of attributed variables.
 | |
| *
 | |
| * Revision 1.28  2005/09/08 22:06:44  rslopes
 | |
| * BEAM for YAP update...
 | |
| *
 | |
| * Revision 1.27  2005/07/06 15:10:04  vsc
 | |
| * improvements to compiler: merged instructions and fixes for ->
 | |
| *
 | |
| * Revision 1.26  2005/01/04 02:50:21  vsc
 | |
| * - allow MegaClauses with blobs
 | |
| * - change Diffs to be thread specific
 | |
| * - include Christian's updates
 | |
| *
 | |
| * Revision 1.25  2004/11/19 17:14:13  vsc
 | |
| * a few fixes for 64 bit compiling.
 | |
| *
 | |
| * Revision 1.24  2004/04/16 19:27:31  vsc
 | |
| * more bug fixes
 | |
| *
 | |
| * Revision 1.23  2004/03/10 14:59:55  vsc
 | |
| * optimise -> for type tests
 | |
| *									 *
 | |
| *									 *
 | |
| *************************************************************************/
 | |
| #ifdef SCCS
 | |
| static char SccsId[] = "%W% %G%";
 | |
| #endif
 | |
| 
 | |
| /*
 | |
|  * This file includes a set of utilities, useful to the several compilation
 | |
|  * modules 
 | |
|  */
 | |
| 
 | |
| #include "Yap.h"
 | |
| #include "Yatom.h"
 | |
| #include "YapHeap.h"
 | |
| #include "compile.h"
 | |
| #include "yapio.h"
 | |
| #if HAVE_STRING_H
 | |
| #include <string.h>
 | |
| #endif
 | |
| 
 | |
| #ifdef DEBUG
 | |
| STATIC_PROTO (void ShowOp, (char *, struct PSEUDO *));
 | |
| #endif /* DEBUG */
 | |
| 
 | |
| /*
 | |
|  * The compiler creates an instruction chain which will be assembled after
 | |
|  * afterwards 
 | |
|  */
 | |
| 
 | |
| 
 | |
| 
 | |
| typedef struct mem_blk {
 | |
|   union {
 | |
|     struct mem_blk *next;
 | |
|     double fill;
 | |
|   } u;
 | |
|   char contents[1];
 | |
| } MemBlk;
 | |
| 
 | |
| #define CMEM_BLK_SIZE (4*4096)
 | |
| #define FIRST_CMEM_BLK_SIZE (16*4096)
 | |
| 
 | |
| static char *
 | |
| AllocCMem (UInt size, struct intermediates *cip)
 | |
| {
 | |
| #if SIZEOF_INT_P==8
 | |
|   size = (size + 7) & ((UInt)-8);
 | |
| #else
 | |
|   size = (size + 3) & ((UInt)0xfffffffc);
 | |
| #endif
 | |
| #if USE_SYSTEM_MALLOC
 | |
|   if (!cip->blks || cip->blk_cur+size > cip->blk_top) {
 | |
|     UInt blksz;
 | |
|     struct mem_blk *p;
 | |
| 
 | |
|     if (size > CMEM_BLK_SIZE)
 | |
|       blksz = size+sizeof(struct mem_blk);
 | |
|     else
 | |
|       blksz = CMEM_BLK_SIZE;
 | |
|     if (!cip->blks) {
 | |
|       CACHE_REGS
 | |
|       if (LOCAL_CMemFirstBlock) {
 | |
| 	p = LOCAL_CMemFirstBlock;
 | |
| 	blksz = LOCAL_CMemFirstBlockSz;
 | |
| 	p->u.next = NULL;
 | |
|       } else {
 | |
| 	if (blksz < FIRST_CMEM_BLK_SIZE)
 | |
| 	  blksz = FIRST_CMEM_BLK_SIZE;
 | |
| 	p = (struct mem_blk *)Yap_AllocCodeSpace(blksz);
 | |
| 	if (!p) {
 | |
| 	  LOCAL_Error_Size = size;
 | |
| 	  save_machine_regs();
 | |
| 	  siglongjmp(cip->CompilerBotch, OUT_OF_HEAP_BOTCH);
 | |
| 	}
 | |
| 	LOCAL_CMemFirstBlock = p;
 | |
| 	LOCAL_CMemFirstBlockSz = blksz;
 | |
|       }
 | |
|     } else {
 | |
|       p = (struct mem_blk *)Yap_AllocCodeSpace(blksz);
 | |
|       if (!p) {
 | |
| 	CACHE_REGS
 | |
| 	LOCAL_Error_Size = size;
 | |
| 	save_machine_regs();
 | |
| 	siglongjmp(cip->CompilerBotch, OUT_OF_HEAP_BOTCH);
 | |
|       }
 | |
|     }
 | |
|     p->u.next = cip->blks;
 | |
|     cip->blks = p;
 | |
|     cip->blk_cur = p->contents;
 | |
|     cip->blk_top = (char *)p+blksz;
 | |
|   }
 | |
|   {
 | |
|     char *out = cip->blk_cur;
 | |
|     cip->blk_cur += size;
 | |
|     return out;
 | |
|   }
 | |
| #else
 | |
|   char *p;
 | |
|   p = cip->freep;
 | |
|   cip->freep += size;
 | |
|   if (ASP <= CellPtr (cip->freep) + 256) {
 | |
|     CACHE_REGS
 | |
|     LOCAL_Error_Size = 256+((char *)cip->freep - (char *)H);
 | |
|     save_machine_regs();
 | |
|     siglongjmp(cip->CompilerBotch, OUT_OF_STACK_BOTCH);
 | |
|   }
 | |
|   return (p);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void
 | |
| Yap_ReleaseCMem (struct intermediates *cip)
 | |
| {
 | |
| #if USE_SYSTEM_MALLOC
 | |
|   CACHE_REGS
 | |
|   struct mem_blk *p = cip->blks;
 | |
|   while (p) {
 | |
|     struct mem_blk *nextp = p->u.next;
 | |
|     if (p != LOCAL_CMemFirstBlock)
 | |
|       Yap_FreeCodeSpace((ADDR)p);
 | |
|     p = nextp;
 | |
|   }
 | |
|   cip->blks = NULL;
 | |
|   if (cip->label_offset &&
 | |
|       cip->label_offset != LOCAL_LabelFirstArray) {
 | |
|     Yap_FreeCodeSpace((ADDR)cip->label_offset);
 | |
|   }
 | |
| #endif
 | |
|   cip->label_offset = NULL;
 | |
| }
 | |
| 
 | |
| char *
 | |
| Yap_AllocCMem (UInt size, struct intermediates *cip)
 | |
| {
 | |
|   return AllocCMem(size, cip);
 | |
| }
 | |
| 
 | |
| static int
 | |
| is_a_test(Term arg, Term mod)
 | |
| {
 | |
|   if (IsVarTerm (arg)) {
 | |
|     return FALSE;
 | |
|   }
 | |
|   if (IsVarTerm (arg) || !IsAtomTerm(mod)) {
 | |
|     return FALSE;
 | |
|   }
 | |
|   if (IsAtomTerm (arg)) {
 | |
|       Atom At = AtomOfTerm (arg);
 | |
|       PredEntry *pe = RepPredProp(PredPropByAtom(At, mod));
 | |
|       if (EndOfPAEntr(pe))
 | |
| 	return FALSE;
 | |
|       return pe->PredFlags & TestPredFlag;
 | |
|   }
 | |
|   if (IsApplTerm (arg)) {
 | |
|     Functor f = FunctorOfTerm (arg);
 | |
| 
 | |
|     if (f == FunctorModule) {
 | |
|       return is_a_test(ArgOfTerm(2,arg), ArgOfTerm(1,arg));
 | |
|     } else if (f == FunctorComma) {
 | |
|       return
 | |
| 	is_a_test(ArgOfTerm(1,arg), mod) &&
 | |
| 	is_a_test(ArgOfTerm(2,arg), mod);
 | |
|     } else {
 | |
|       PredEntry *pe = RepPredProp(PredPropByFunc(f, mod));
 | |
| 
 | |
|       if (EndOfPAEntr(pe))
 | |
| 	return FALSE;
 | |
|       if (pe->PredFlags & AsmPredFlag) {
 | |
| 	int op = pe->PredFlags & 0x7f;
 | |
| 	if (op >= _atom && op <= _eq) {
 | |
| 	  return TRUE;
 | |
| 	}
 | |
| 	return FALSE;
 | |
|       }      
 | |
|       return pe->PredFlags & (TestPredFlag|BinaryPredFlag);
 | |
|     }
 | |
|   }
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| int
 | |
| Yap_is_a_test_pred (Term arg, Term mod)
 | |
| {
 | |
|   return is_a_test(arg, mod);
 | |
| }
 | |
| 
 | |
| void
 | |
| Yap_emit (compiler_vm_op o, Int r1, CELL r2, struct intermediates *cip)
 | |
| {
 | |
|   PInstr *p;
 | |
|   p = (PInstr *) AllocCMem (sizeof (*p), cip);
 | |
|   p->op = o;
 | |
|   p->rnd1 = r1;
 | |
|   p->rnd2 = r2;
 | |
|   p->nextInst = NULL;
 | |
|   if (cip->cpc == NIL) {
 | |
|     cip->cpc = cip->CodeStart = p;
 | |
|   } else {
 | |
|     cip->cpc->nextInst = p;
 | |
|     cip->cpc = p;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void
 | |
| Yap_emit_3ops (compiler_vm_op o, CELL r1, CELL r2, CELL r3, struct intermediates *cip)
 | |
| {
 | |
|   PInstr *p;
 | |
|   p = (PInstr *) AllocCMem (sizeof (*p)+sizeof(CELL), cip);
 | |
|   p->op = o;
 | |
|   p->rnd1 = r1;
 | |
|   p->rnd2 = r2;
 | |
|   p->rnd3 = r3;
 | |
|   p->nextInst = NIL;
 | |
|   if (cip->cpc == NIL)
 | |
|     cip->cpc = cip->CodeStart = p;
 | |
|   else
 | |
|     {
 | |
|       cip->cpc->nextInst = p;
 | |
|       cip->cpc = p;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void
 | |
| Yap_emit_4ops (compiler_vm_op o, CELL r1, CELL r2, CELL r3, CELL r4, struct intermediates *cip)
 | |
| {
 | |
|   PInstr *p;
 | |
|   p = (PInstr *) AllocCMem (sizeof (*p)+2*sizeof(CELL), cip);
 | |
|   p->op = o;
 | |
|   p->rnd1 = r1;
 | |
|   p->rnd2 = r2;
 | |
|   p->rnd3 = r3;
 | |
|   p->rnd4 = r4;
 | |
|   p->nextInst = NIL;
 | |
|   if (cip->cpc == NIL)
 | |
|     cip->cpc = cip->CodeStart = p;
 | |
|   else
 | |
|     {
 | |
|       cip->cpc->nextInst = p;
 | |
|       cip->cpc = p;
 | |
|     }
 | |
| }
 | |
| 
 | |
| CELL *
 | |
| Yap_emit_extra_size (compiler_vm_op o, CELL r1, int size, struct intermediates *cip)
 | |
| {
 | |
|   PInstr *p;
 | |
|   p = (PInstr *) AllocCMem (sizeof (*p) + size - CellSize, cip);
 | |
|   p->op = o;
 | |
|   p->rnd1 = r1;
 | |
|   p->nextInst = NIL;
 | |
|   if (cip->cpc == NIL)
 | |
|     cip->cpc = cip->CodeStart = p;
 | |
|   else
 | |
|     {
 | |
|       cip->cpc->nextInst = p;
 | |
|       cip->cpc = p;
 | |
|     }
 | |
|   return p->arnds;
 | |
| }
 | |
| 
 | |
| static void
 | |
| bip_name(Int op, char *s)
 | |
| {
 | |
|   switch (op) {
 | |
|   case _atom:
 | |
|     strcpy(s,"atom");
 | |
|     break;
 | |
|   case _atomic:
 | |
|     strcpy(s,"atomic");
 | |
|     break;
 | |
|   case _integer:
 | |
|     strcpy(s,"integer");
 | |
|     break;
 | |
|   case _nonvar:
 | |
|     strcpy(s,"nonvar");
 | |
|     break;
 | |
|   case _number:
 | |
|     strcpy(s,"number");
 | |
|     break;
 | |
|   case _var:
 | |
|     strcpy(s,"var");
 | |
|     break;
 | |
|   case _cut_by:
 | |
|     strcpy(s,"cut_by");
 | |
|     break;
 | |
|   case _db_ref:
 | |
|     strcpy(s,"db_ref");
 | |
|     break;
 | |
|   case _compound:
 | |
|     strcpy(s,"compound");
 | |
|     break;
 | |
|   case _float:
 | |
|     strcpy(s,"float");
 | |
|     break;
 | |
|   case _primitive:
 | |
|     strcpy(s,"primitive");
 | |
|     break;
 | |
|   case _equal:
 | |
|     strcpy(s,"equal");
 | |
|     break;
 | |
|   case _dif:
 | |
|     strcpy(s,"dif");
 | |
|     break;
 | |
|   case _eq:
 | |
|     strcpy(s,"eq");
 | |
|     break;
 | |
|   case _functor:
 | |
|     strcpy(s,"functor");
 | |
|     break;
 | |
|   case _plus:
 | |
|     strcpy(s,"plus");
 | |
|     break;
 | |
|   case _minus:
 | |
|     strcpy(s,"minus");
 | |
|     break;
 | |
|   case _times:
 | |
|     strcpy(s,"times");
 | |
|     break;
 | |
|   case _div:
 | |
|     strcpy(s,"div");
 | |
|     break;
 | |
|   case _and:
 | |
|     strcpy(s,"and");
 | |
|     break;
 | |
|   case _or:
 | |
|     strcpy(s,"or");
 | |
|     break;
 | |
|   case _sll:
 | |
|     strcpy(s,"sll");
 | |
|     break;
 | |
|   case _slr:
 | |
|     strcpy(s,"slr");
 | |
|     break;
 | |
|   case _arg:
 | |
|     strcpy(s,"arg");
 | |
|     break;
 | |
|   default:
 | |
|     strcpy(s,"");
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void
 | |
| Yap_bip_name(Int op, char *s) {
 | |
|   bip_name(op,s);
 | |
| }
 | |
| 
 | |
| #ifdef DEBUG
 | |
| 
 | |
| static void
 | |
| write_address(CELL address)
 | |
| {
 | |
|   if (address < (CELL)AtomBase) {
 | |
|     Yap_DebugErrorPutc('L');
 | |
|     Yap_DebugPlWrite(MkIntTerm (address));
 | |
|   } else if (address == (CELL) FAILCODE) {
 | |
|     Yap_DebugPlWrite (MkAtomTerm (AtomFail));
 | |
|   } else {
 | |
|     char buf[32], *p = buf;
 | |
| 
 | |
| #if HAVE_SNPRINTF
 | |
|     snprintf(buf,32,"%p",(void *)address);
 | |
| #else
 | |
|     sprintf(buf,"%p",(void *)address);
 | |
| #endif
 | |
|     p[31] = '\0'; /* so that I don't have to worry */
 | |
|     Yap_DebugErrorPutc('0');
 | |
|     Yap_DebugErrorPutc('x');
 | |
|     while (*p != '\0') {
 | |
|       Yap_DebugErrorPutc(*p++);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void
 | |
| write_functor(Functor f)
 | |
| {
 | |
|   if (IsExtensionFunctor(f)) {
 | |
|     if (f == FunctorDBRef) {
 | |
|       Yap_DebugPlWrite(MkAtomTerm(AtomDBREF));
 | |
|     } else if (f == FunctorLongInt) {
 | |
|       Yap_DebugPlWrite(MkAtomTerm(AtomLONGINT));
 | |
|     } else if (f == FunctorDouble) {
 | |
|       Yap_DebugPlWrite(MkAtomTerm(AtomDOUBLE));
 | |
|     }
 | |
|   } else {
 | |
|     Yap_DebugPlWrite(MkAtomTerm(NameOfFunctor (f)));
 | |
|     Yap_DebugErrorPutc ('/');
 | |
|     Yap_DebugPlWrite(MkIntTerm(ArityOfFunctor (f)));
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void
 | |
| ShowOp (char *f, struct PSEUDO *cpc)
 | |
| {
 | |
|   char ch;
 | |
|   Int arg = cpc->rnd1;
 | |
|   Int rn = cpc->rnd2;
 | |
|   CELL *cptr = cpc->arnds;
 | |
| 
 | |
|   while ((ch = *f++) != 0)
 | |
|     {
 | |
|       if (ch == '%')
 | |
| 	switch (ch = *f++)
 | |
| 	  {
 | |
| #ifdef BEAM
 | |
| 	case '1':
 | |
| 		Yap_DebugPlWrite(MkIntTerm(rn));
 | |
| 		break;
 | |
| 	case '4':
 | |
| 		Yap_DebugPlWrite(MkIntTerm(arg));
 | |
| 		break;
 | |
| #endif
 | |
| 	  case 'a':
 | |
| 	  case 'n':
 | |
| 	    Yap_DebugPlWrite ((Term) arg);
 | |
| 	    break;
 | |
| 	  case 'b':
 | |
| 	    /* write a variable bitmap for a call */
 | |
| 	    {
 | |
| 	      int max = arg/(8*sizeof(CELL)), i;
 | |
| 	      CELL *ptr = cptr;
 | |
| 	      for (i = 0; i <= max; i++) {
 | |
| 		Yap_DebugPlWrite(MkIntegerTerm((Int)(*ptr++)));
 | |
| 	      }
 | |
| 	    }
 | |
| 	    break;		
 | |
| 	  case 'l':
 | |
| 	    write_address (arg);
 | |
| 	    break;
 | |
| 	  case 'B':
 | |
| 	    {
 | |
| 	      char s[32];
 | |
| 
 | |
| 	      bip_name(rn,s);
 | |
| 	      Yap_DebugPlWrite (MkAtomTerm(Yap_LookupAtom(s)));
 | |
| 	    }
 | |
| 	    break;
 | |
| 	  case 'd':
 | |
| 	    Yap_DebugPlWrite (MkIntegerTerm (arg));
 | |
| 	    break;
 | |
| 	  case 'z':
 | |
| 	    Yap_DebugPlWrite (MkIntTerm (cpc->rnd3));
 | |
| 	    break;
 | |
| 	  case 'v':
 | |
| 	    {
 | |
| 	      Ventry *v = (Ventry *) arg;
 | |
| 	      Yap_DebugErrorPutc (v->KindOfVE == PermVar ? 'Y' : 'X');
 | |
| 	      Yap_DebugPlWrite (MkIntTerm ((v->NoOfVE) & MaskVarAdrs));
 | |
| 	    }
 | |
| 	    break;	
 | |
| 	  case 'N':
 | |
| 	    {
 | |
| 	      Ventry *v;
 | |
| 
 | |
| 	      cpc = cpc->nextInst;
 | |
| 	      arg = cpc->rnd1;
 | |
| 	      v = (Ventry *) arg;
 | |
| 	      Yap_DebugErrorPutc (v->KindOfVE == PermVar ? 'Y' : 'X');
 | |
| 	      Yap_DebugPlWrite (MkIntTerm ((v->NoOfVE) & MaskVarAdrs));
 | |
| 	    }
 | |
| 	    break;
 | |
| 	  case 'm':
 | |
| 	    Yap_DebugPlWrite (MkAtomTerm ((Atom) arg));
 | |
| 	    Yap_DebugErrorPutc ('/');
 | |
| 	    Yap_DebugPlWrite (MkIntTerm (rn));
 | |
| 	    break;
 | |
| 	  case 'p':
 | |
| 	    {
 | |
| 	      PredEntry *p = RepPredProp ((Prop) arg);
 | |
| 	      Functor f = p->FunctorOfPred;
 | |
| 	      UInt arity = p->ArityOfPE;
 | |
| 	      Term mod;
 | |
| 
 | |
| 	      if (p->ModuleOfPred)
 | |
| 		mod = p->ModuleOfPred;
 | |
| 	      else
 | |
| 		mod = TermProlog;
 | |
| 	      Yap_DebugPlWrite (mod);
 | |
| 	      Yap_DebugErrorPutc (':');
 | |
| 	      if (arity == 0)
 | |
| 		Yap_DebugPlWrite (MkAtomTerm ((Atom)f));
 | |
| 	      else
 | |
| 		Yap_DebugPlWrite (MkAtomTerm (NameOfFunctor (f)));
 | |
| 	      Yap_DebugErrorPutc ('/');
 | |
| 	      Yap_DebugPlWrite (MkIntTerm (arity));
 | |
| 	    }
 | |
| 	    break;
 | |
| 	  case 'P':
 | |
| 	    {
 | |
| 	      PredEntry *p = RepPredProp((Prop) rn);
 | |
| 	      Functor f = p->FunctorOfPred;
 | |
| 	      UInt arity = p->ArityOfPE;
 | |
| 	      Term mod = TermProlog;
 | |
| 
 | |
| 	      if (p->ModuleOfPred) mod = p->ModuleOfPred;
 | |
| 	      Yap_DebugPlWrite (mod);
 | |
| 	      Yap_DebugErrorPutc (':');
 | |
| 	      if (arity == 0)
 | |
| 		Yap_DebugPlWrite (MkAtomTerm ((Atom)f));
 | |
| 	      else
 | |
| 		Yap_DebugPlWrite (MkAtomTerm (NameOfFunctor (f)));
 | |
| 	      Yap_DebugErrorPutc ('/');
 | |
| 	      Yap_DebugPlWrite (MkIntTerm (arity));
 | |
| 	    }
 | |
| 	    break;
 | |
| 	  case 'f':
 | |
| 	    write_functor((Functor)arg);
 | |
| 	    break;
 | |
| 	  case 'r':
 | |
| 	    Yap_DebugErrorPutc ('A');
 | |
| 	    Yap_DebugPlWrite (MkIntTerm (rn));
 | |
| 	    break;
 | |
| 	  case 'h':
 | |
| 	    {
 | |
| 	      CELL my_arg = *cptr++;
 | |
| 	      write_address(my_arg);
 | |
| 	    }
 | |
| 	    break;
 | |
| 	  case 'g':
 | |
| 	    write_address(arg);
 | |
| 	    break;
 | |
| 	  case 'i':
 | |
| 	    write_address (arg);
 | |
| 	    break;
 | |
| 	  case 'j':
 | |
| 	    {
 | |
| 	      Functor fun = (Functor)*cptr++;
 | |
| 	      if (IsExtensionFunctor(fun)) {
 | |
| 		if (fun == FunctorDBRef) {
 | |
| 		  Yap_DebugPlWrite(MkAtomTerm(AtomDBREF));
 | |
| 		} else if (fun == FunctorLongInt) {
 | |
| 		  Yap_DebugPlWrite(MkAtomTerm(AtomLONGINT));
 | |
| 		} else if (fun == FunctorDouble) {
 | |
| 		  Yap_DebugPlWrite(MkAtomTerm(AtomDOUBLE));
 | |
| 		}
 | |
| 	      } else {
 | |
| 		Yap_DebugPlWrite (MkAtomTerm(NameOfFunctor(fun)));
 | |
| 		Yap_DebugErrorPutc ('/');
 | |
| 		Yap_DebugPlWrite (MkIntTerm(ArityOfFunctor(fun)));
 | |
| 	      }
 | |
| 	    }
 | |
| 	    break;
 | |
| 	  case 'O':
 | |
| 	    Yap_DebugPlWrite(AbsAppl(cptr));
 | |
| 	    break;
 | |
| 	  case 'x':
 | |
| 	    Yap_DebugPlWrite (MkIntTerm (rn >> 1));
 | |
| 	    Yap_DebugErrorPutc ('\t');
 | |
| 	    Yap_DebugPlWrite (MkIntTerm (rn & 1));
 | |
| 	    break;
 | |
| 	  case 'w':
 | |
| 	    Yap_DebugPlWrite (arg);
 | |
| 	    break;
 | |
| 	  case 'o':
 | |
| 	    Yap_DebugPlWrite ((Term) * cptr++);
 | |
| 	  case 'c':
 | |
| 	    {
 | |
| 	      int i;
 | |
| 	      CELL *ptr = (CELL *)cptr[0];
 | |
| 	      for (i = 0; i < arg; ++i) {
 | |
| 		CELL my_arg;
 | |
| 		Yap_DebugErrorPutc('\t');
 | |
| 		if (*ptr) {
 | |
| 		  Yap_DebugPlWrite ((Term) *ptr++);
 | |
| 		} else {
 | |
| 		  Yap_DebugPlWrite (MkIntTerm (0));
 | |
| 		  ptr++;
 | |
| 		}
 | |
| 		Yap_DebugErrorPutc ('\t');
 | |
| 		my_arg = *ptr++;
 | |
| 		write_address (my_arg);
 | |
| 		if (i+1 < arg)
 | |
| 		  Yap_DebugErrorPutc ('\n');
 | |
| 	      }
 | |
| 	    }
 | |
| 	    break;
 | |
| 	  case 'e':
 | |
| 	    {
 | |
| 	      int i;
 | |
| 	      CELL *ptr = (CELL *)cptr[0];
 | |
| 	      for (i = 0; i < arg; ++i)	{
 | |
| 		CELL my_arg = ptr[0], lbl = ptr[1];
 | |
| 		Yap_DebugErrorPutc('\t');
 | |
| 		if (my_arg) {
 | |
| 		  write_functor((Functor)my_arg);
 | |
| 		} else {
 | |
| 		  Yap_DebugPlWrite(MkIntTerm (0));
 | |
| 		}
 | |
| 		Yap_DebugErrorPutc('\t');
 | |
| 		write_address(lbl);
 | |
| 		ptr += 2;
 | |
| 		if (i+1 < arg)
 | |
| 		  Yap_DebugErrorPutc('\n');
 | |
| 	      }
 | |
| 	    }
 | |
| 	    break;
 | |
| 	  default:
 | |
| 	    Yap_DebugErrorPutc ('%');
 | |
| 	    Yap_DebugErrorPutc (ch);
 | |
| 	  }
 | |
|       else
 | |
| 	Yap_DebugErrorPutc (ch);
 | |
|     }
 | |
|   Yap_DebugErrorPutc ('\n');
 | |
| }
 | |
| 
 | |
| static char *opformat[] =
 | |
| {
 | |
|   "nop",
 | |
|   "get_var\t\t%v,%r",
 | |
|   "put_var\t\t%v,%r",
 | |
|   "get_val\t\t%v,%r",
 | |
|   "put_val\t\t%v,%r",
 | |
|   "get_atom\t%a,%r",
 | |
|   "put_atom\t%a,%r",
 | |
|   "get_num\t\t%n,%r",
 | |
|   "put_num\t\t%n,%r",
 | |
|   "get_float\t\t%w,%r",
 | |
|   "put_float\t\t%w,%r",
 | |
|   "get_dbterm\t%w,%r",
 | |
|   "put_dbterm\t%w,%r",
 | |
|   "get_longint\t\t%w,%r",
 | |
|   "put_longint\t\t%w,%r",
 | |
|   "get_bigint\t\t%l,%r",
 | |
|   "put_bigint\t\t%l,%r",
 | |
|   "get_list\t%r",
 | |
|   "put_list\t%r",
 | |
|   "get_struct\t%f,%r",
 | |
|   "put_struct\t%f,%r",
 | |
|   "put_unsafe\t%v,%r",
 | |
|   "unify_var\t%v",
 | |
|   "write_var\t%v",
 | |
|   "unify_val\t%v",
 | |
|   "write_val\t%v",
 | |
|   "unify_atom\t%a",
 | |
|   "write_atom\t%a",
 | |
|   "unify_num\t%n",
 | |
|   "write_num\t%n",
 | |
|   "unify_float\t%w",
 | |
|   "write_float\t%w",
 | |
|   "unify_dbterm\t%w",
 | |
|   "write_dbterm\t%w",
 | |
|   "unify_longint\t%w",
 | |
|   "write_longint\t%w",
 | |
|   "unify_bigint\t%l",
 | |
|   "write_bigint\t%l",
 | |
|   "unify_list",
 | |
|   "write_list",
 | |
|   "unify_struct\t%f",
 | |
|   "write_struct\t%f",
 | |
|   "write_unsafe\t%v",
 | |
|   "unify_local\t%v",
 | |
|   "write local\t%v",
 | |
|   "unify_last_list",
 | |
|   "write_last_list",
 | |
|   "unify_last_struct\t%f",
 | |
|   "write_last_struct\t%f",
 | |
|   "unify_last_var\t%v",
 | |
|   "unify_last_val\t%v",
 | |
|   "unify_last_local\t%v",
 | |
|   "unify_last_atom\t%a",
 | |
|   "unify_last_num\t%n",
 | |
|   "unify_last_float\t%w", 
 | |
|   "unify_last_dbterm\t%w",
 | |
|   "unify_last_longint\t%w",
 | |
|   "unify_last_bigint\t%l",
 | |
|   "ensure_space",
 | |
|   "native_code",
 | |
|   "function_to_var\t%v,%B",
 | |
|   "function_to_val\t%v,%B",
 | |
|   "function_to_0\t%B",
 | |
|   "align_float",
 | |
|   "fail",
 | |
|   "cut",
 | |
|   "cutexit",
 | |
|   "allocate",
 | |
|   "deallocate",
 | |
|   "try_me_else\t\t%l\t%x",
 | |
|   "jump\t\t%l",
 | |
|   "jump\t\t%l",
 | |
|   "proceed",
 | |
|   "call\t\t%p,%d,%z",
 | |
|   "execute\t\t%p",
 | |
|   "sys\t\t%p",
 | |
|   "%l:",
 | |
|   "name\t\t%m,%d",
 | |
|   "pop\t\t%l",
 | |
|   "retry_me_else\t\t%l\t%x",
 | |
|   "trust_me_else_fail\t%x",
 | |
|   "either_me\t\t%l,%d,%z",
 | |
|   "or_else\t\t%l,%z",
 | |
|   "or_last",
 | |
|   "push_or",
 | |
|   "pushpop_or",
 | |
|   "pop_or",
 | |
|   "save_by\t\t%v",
 | |
|   "commit_by\t\t%v",
 | |
|   "patch_by\t\t%v",
 | |
|   "try\t\t%g\t%x",
 | |
|   "retry\t\t%g\t%x",
 | |
|   "trust\t\t%g\t%x",
 | |
|   "try_in\t\t%g\t%x",
 | |
|   "jump_if_var\t\t%g",
 | |
|   "jump_if_nonvar\t\t%g",
 | |
|   "cache_arg\t%r",
 | |
|   "cache_sub_arg\t%d",
 | |
|   "user_index",
 | |
|   "switch_on_type\t%h\t%h\t%h\t%h",
 | |
|   "switch_on_constant\t%i\n%c",
 | |
|   "if_constant\t%i\n%c",
 | |
|   "switch_on_functor\t%i\n%e",
 | |
|   "if_functor\t%i\n%e",
 | |
|   "if_not_then\t%i\t%h\t%h\t%h",
 | |
|   "index_on_dbref",
 | |
|   "index_on_blob",
 | |
|   "index_on_long",
 | |
|   "check_var\t %r",
 | |
|   "save_pair\t%v",
 | |
|   "save_appl\t%v",
 | |
|   "pvar_bitmap\t%l,%b",
 | |
|   "pvar_live_regs\t%l,%b",
 | |
|   "fetch_reg1_reg2\t%N,%N",
 | |
|   "fetch_constant_reg\t%l,%N",
 | |
|   "fetch_reg_constant\t%l,%N",
 | |
|   "fetch_integer_reg\t%d,%N",
 | |
|   "fetch_reg_integer\t%d,%N",
 | |
|   "enter_profiling\t\t%g",
 | |
|   "retry_profiled\t\t%g",
 | |
|   "count_call_op\t\t%g",
 | |
|   "count_retry_op\t\t%g",
 | |
|   "restore_temps\t\t%l",
 | |
|   "restore_temps_and_skip\t\t%l",
 | |
|   "enter_lu",
 | |
|   "empty_call\t\t%l,%d",
 | |
| #ifdef YAPOR
 | |
|   "sync",
 | |
| #endif /* YAPOR */
 | |
| #ifdef TABLING
 | |
|   "table_new_answer",
 | |
|   "table_try_single\t%g\t%x",
 | |
| #endif /* TABLING */
 | |
| #ifdef TABLING_INNER_CUTS
 | |
|   "clause_with_cut",
 | |
| #endif /* TABLING_INNER_CUTS */
 | |
| #ifdef BEAM
 | |
|   "run_op %1,%4",
 | |
|   "body_op %1",
 | |
|   "endgoal_op",
 | |
|   "try_me_op %1,%4",
 | |
|   "retry_me_op %1,%4",
 | |
|   "trust_me_op %1,%4",
 | |
|   "only_1_clause_op %1,%4",
 | |
|   "create_first_box_op %1,%4",
 | |
|   "create_box_op %1,%4",
 | |
|   "create_last_box_op %1,%4",
 | |
|   "remove_box_op %1,%4",
 | |
|   "remove_last_box_op %1,%4",
 | |
|   "prepare_tries",
 | |
|   "std_base_op %1,%4",
 | |
|   "direct_safe_call",
 | |
|   "skip_while_var_op",
 | |
|   "wait_while_var_op",
 | |
|   "force_wait_op",
 | |
|   "write_op",
 | |
|   "is_op",
 | |
|   "equal_op",
 | |
|   "exit",
 | |
| #endif
 | |
|   "fetch_args_for_bccall\t%v",
 | |
|   "binary_cfunc\t\t%v,%P",
 | |
|   "blob\t%O",
 | |
|   "label_control\t"
 | |
| #ifdef SFUNC
 | |
|   ,
 | |
|   "get_s_f_op\t%f,%r",
 | |
|   "put_s_f_op\t%f,%r",
 | |
|   "unify_s_f_op\t%f",
 | |
|   "write_s_f_op\t%f",
 | |
|   "unify_s_var\t%v,%r",
 | |
|   "write_s_var\t%v,%r",
 | |
|   "unify_s_val\t%v,%r",
 | |
|   "write_s_val\t%v,%r",
 | |
|   "unify_s_a\t%a,%r",
 | |
|   "write_s_a\t%a,%r",
 | |
|   "get_s_end",
 | |
|   "put_s_end",
 | |
|   "unify_s_end",
 | |
|   "write_s_end"
 | |
| #endif
 | |
| };
 | |
| 
 | |
| 
 | |
| void
 | |
| Yap_ShowCode (struct intermediates *cint)
 | |
| {
 | |
|   CACHE_REGS
 | |
|   CELL *oldH = H;
 | |
|   struct PSEUDO *cpc;
 | |
| 
 | |
|   cpc = cint->CodeStart;
 | |
|   /* MkIntTerm and friends may build terms in the global stack */
 | |
|   H = (CELL *)cint->freep;
 | |
|   while (cpc) {
 | |
|     compiler_vm_op ic = cpc->op;
 | |
|     if (ic != nop_op) {
 | |
|       ShowOp (opformat[ic], cpc);
 | |
|     }
 | |
|     cpc = cpc->nextInst;
 | |
|   }
 | |
|   Yap_DebugErrorPutc ('\n');
 | |
|   H = oldH;
 | |
| }
 | |
| 
 | |
| #endif /* DEBUG */
 | |
| 
 |