/************************************************************************* * * * 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 * * Last rev: * * mods: * * comments: some useful routines for YAP's compiler * * * *************************************************************************/ #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 "Heap.h" #include "compile.h" #include "yapio.h" #if HAVE_STRING_H #include #endif #ifdef DEBUG STATIC_PROTO (void ShowOp, (char *)); #endif /* DEBUG */ /* * The compiler creates an instruction chain which will be assembled after * afterwards */ #ifdef DEBUG static Int arg, rn; static compiler_vm_op ic; static CELL *cptr; char Yap_Option[20]; YP_FILE *Yap_logfile; #endif static char * AllocCMem (int size) { char *p; p = freep; #if SIZEOF_INT_P==8 size = (size + 7) & 0xfffffffffffffff8L; #else size = (size + 3) & 0xfffffffcL; #endif freep += size; if (ASP <= CellPtr (freep) + 256) { Yap_Error_Size = 256+((char *)freep - (char *)H); save_machine_regs(); longjmp(Yap_CompilerBotch,3); } return (p); } char * Yap_AllocCMem (int size) { return(AllocCMem(size)); } int Yap_is_a_test_pred (Term arg, SMALLUNSGN mod) { if (IsVarTerm (arg)) return (FALSE); else if (IsAtomTerm (arg)) { Atom At = AtomOfTerm (arg); PredEntry *pe = RepPredProp(PredPropByAtom(At, mod)); if (EndOfPAEntr(pe)) return (FALSE); return (pe->PredFlags & TestPredFlag); } else if (IsApplTerm (arg)) { Functor f = FunctorOfTerm (arg); PredEntry *pe = RepPredProp(PredPropByFunc(f, mod)); if (EndOfPAEntr(pe)) return (FALSE); return (pe->PredFlags & TestPredFlag); } else { return (FALSE); } } void Yap_emit (compiler_vm_op o, Int r1, CELL r2) { PInstr *p; p = (PInstr *) AllocCMem (sizeof (*p)); p->op = o; p->rnd1 = r1; p->rnd2 = r2; p->nextInst = NULL; if (cpc == NIL) { cpc = CodeStart = p; } else { cpc->nextInst = p; cpc = p; } } void Yap_emit_3ops (compiler_vm_op o, CELL r1, CELL r2, CELL r3) { PInstr *p; p = (PInstr *) AllocCMem (sizeof (*p)+sizeof(CELL)); p->op = o; p->rnd1 = r1; p->rnd2 = r2; p->rnd3 = r3; p->nextInst = NIL; if (cpc == NIL) cpc = CodeStart = p; else { cpc->nextInst = p; cpc = p; } } void Yap_emit_4ops (compiler_vm_op o, CELL r1, CELL r2, CELL r3, CELL r4) { PInstr *p; p = (PInstr *) AllocCMem (sizeof (*p)+2*sizeof(CELL)); p->op = o; p->rnd1 = r1; p->rnd2 = r2; p->rnd3 = r3; p->rnd4 = r4; p->nextInst = NIL; if (cpc == NIL) cpc = CodeStart = p; else { cpc->nextInst = p; cpc = p; } } CELL * Yap_emit_extra_size (compiler_vm_op o, CELL r1, int size) { PInstr *p; p = (PInstr *) AllocCMem (sizeof (*p) + size - CellSize); p->op = o; p->rnd1 = r1; p->nextInst = NIL; if (cpc == NIL) cpc = CodeStart = p; else { cpc->nextInst = p; 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_DebugPutc(Yap_c_error_stream,'L'); Yap_plwrite (MkIntTerm (address), Yap_DebugPutc, 0); } else if (address == (CELL) FAILCODE) { Yap_plwrite (MkAtomTerm (AtomFail), Yap_DebugPutc, 0); } else { char buf[32], *p = buf; #if HAVE_SNPRINTF snprintf(buf,32,"%x",address); #else snprintf(buf,"%x",address); #endif p[31] = '\0'; /* so that I don't have to worry */ Yap_DebugPutc(Yap_c_error_stream,'0'); Yap_DebugPutc(Yap_c_error_stream,'x'); while (*p != '\0') { Yap_DebugPutc(Yap_c_error_stream,*p++); } } } static void write_functor(Functor f) { if (IsExtensionFunctor(f)) { if (f == FunctorDBRef) { Yap_plwrite(MkAtomTerm(Yap_LookupAtom("DBRef")), Yap_DebugPutc, 0); } else if (f == FunctorLongInt) { Yap_plwrite(MkAtomTerm(Yap_LookupAtom("LongInt")), Yap_DebugPutc, 0); } else if (f == FunctorDouble) { Yap_plwrite(MkAtomTerm(Yap_LookupAtom("Double")), Yap_DebugPutc, 0); } } else { Yap_plwrite(MkAtomTerm(NameOfFunctor (f)), Yap_DebugPutc, 0); Yap_DebugPutc (Yap_c_error_stream,'/'); Yap_plwrite(MkIntTerm(ArityOfFunctor (f)), Yap_DebugPutc, 0); } } static void ShowOp (char *f) { char ch; while ((ch = *f++) != 0) { if (ch == '%') switch (ch = *f++) { case 'a': case 'n': Yap_plwrite ((Term) arg, Yap_DebugPutc, 0); 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_plwrite(MkIntegerTerm((Int)(*ptr++)), Yap_DebugPutc, 0); } } break; case 'l': write_address (arg); break; case 'B': { char s[32]; bip_name(rn,s); Yap_plwrite (MkAtomTerm(Yap_LookupAtom(s)), Yap_DebugPutc, 0); } break; case 'd': Yap_plwrite (MkIntTerm (rn), Yap_DebugPutc, 0); break; case 'z': Yap_plwrite (MkIntTerm (cpc->rnd3), Yap_DebugPutc, 0); break; case 'v': { Ventry *v = (Ventry *) arg; Yap_DebugPutc (Yap_c_error_stream,v->KindOfVE == PermVar ? 'Y' : 'X'); Yap_plwrite (MkIntTerm ((v->NoOfVE) & MaskVarAdrs), Yap_DebugPutc, 0); } break; case 'N': { Ventry *v; cpc = cpc->nextInst; arg = cpc->rnd1; v = (Ventry *) arg; Yap_DebugPutc (Yap_c_error_stream,v->KindOfVE == PermVar ? 'Y' : 'X'); Yap_plwrite (MkIntTerm ((v->NoOfVE) & MaskVarAdrs), Yap_DebugPutc, 0); } break; case 'm': Yap_plwrite (MkAtomTerm ((Atom) arg), Yap_DebugPutc, 0); Yap_DebugPutc (Yap_c_error_stream,'/'); Yap_plwrite (MkIntTerm (rn), Yap_DebugPutc, 0); break; case 'p': { PredEntry *p = RepPredProp ((Prop) arg); Functor f = p->FunctorOfPred; UInt arity = p->ArityOfPE; SMALLUNSGN mod = 0; if (p->ModuleOfPred) mod = IntOfTerm(p->ModuleOfPred); Yap_plwrite (ModuleName[mod], Yap_DebugPutc, 0); Yap_DebugPutc (Yap_c_error_stream,':'); if (arity == 0) Yap_plwrite (MkAtomTerm ((Atom)f), Yap_DebugPutc, 0); else Yap_plwrite (MkAtomTerm (NameOfFunctor (f)), Yap_DebugPutc, 0); Yap_DebugPutc (Yap_c_error_stream,'/'); Yap_plwrite (MkIntTerm (arity), Yap_DebugPutc, 0); } break; case 'P': { PredEntry *p = RepPredProp((Prop) rn); Functor f = p->FunctorOfPred; UInt arity = p->ArityOfPE; SMALLUNSGN mod = 0; if (p->ModuleOfPred) mod = IntOfTerm(p->ModuleOfPred); Yap_plwrite (ModuleName[mod], Yap_DebugPutc, 0); Yap_DebugPutc (Yap_c_error_stream,':'); if (arity == 0) Yap_plwrite (MkAtomTerm ((Atom)f), Yap_DebugPutc, 0); else Yap_plwrite (MkAtomTerm (NameOfFunctor (f)), Yap_DebugPutc, 0); Yap_DebugPutc (Yap_c_error_stream,'/'); Yap_plwrite (MkIntTerm (arity), Yap_DebugPutc, 0); } break; case 'f': write_functor((Functor)arg); break; case 'r': Yap_DebugPutc (Yap_c_error_stream,'A'); Yap_plwrite (MkIntTerm (rn), Yap_DebugPutc, 0); 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_plwrite(MkAtomTerm(Yap_LookupAtom("DBRef")), Yap_DebugPutc, 0); } else if (fun == FunctorLongInt) { Yap_plwrite(MkAtomTerm(Yap_LookupAtom("LongInt")), Yap_DebugPutc, 0); } else if (fun == FunctorDouble) { Yap_plwrite(MkAtomTerm(Yap_LookupAtom("Double")), Yap_DebugPutc, 0); } } else { Yap_plwrite (MkAtomTerm(NameOfFunctor(fun)), Yap_DebugPutc, 0); Yap_DebugPutc (Yap_c_error_stream,'/'); Yap_plwrite (MkIntTerm(ArityOfFunctor(fun)), Yap_DebugPutc, 0); } } break; case 'O': Yap_plwrite(AbsAppl(cptr), Yap_DebugPutc, 0); break; case 'x': Yap_plwrite (MkIntTerm (rn >> 1), Yap_DebugPutc, 0); Yap_DebugPutc (Yap_c_error_stream,'\t'); Yap_plwrite (MkIntTerm (rn & 1), Yap_DebugPutc, 0); break; case 'o': Yap_plwrite ((Term) * cptr++, Yap_DebugPutc, 0); case 'c': { int i; CELL *ptr = (CELL *)cptr[0]; for (i = 0; i < arg; ++i) { CELL my_arg; Yap_DebugPutc(Yap_c_error_stream,'\t'); if (*ptr) { Yap_plwrite ((Term) *ptr++, Yap_DebugPutc, 0); } else { Yap_plwrite (MkIntTerm (0), Yap_DebugPutc, 0); ptr++; } Yap_DebugPutc (Yap_c_error_stream,'\t'); my_arg = *ptr++; write_address (my_arg); if (i+1 < arg) Yap_DebugPutc (Yap_c_error_stream,'\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_DebugPutc(Yap_c_error_stream,'\t'); if (my_arg) { write_functor((Functor)my_arg); } else { Yap_plwrite(MkIntTerm (0), Yap_DebugPutc, 0); } Yap_DebugPutc(Yap_c_error_stream,'\t'); write_address(lbl); ptr += 2; if (i+1 < arg) Yap_DebugPutc(Yap_c_error_stream,'\n'); } } break; default: Yap_DebugPutc (Yap_c_error_stream,'%'); Yap_DebugPutc (Yap_c_error_stream,ch); } else Yap_DebugPutc (Yap_c_error_stream,ch); } Yap_DebugPutc (Yap_c_error_stream,'\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%l,%r", "put_float\t\t%l,%r", "get_longint\t\t%l,%r", "put_longint\t\t%l,%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%l", "write_float\t%l", "unify_longint\t%l", "write_longint\t%l", "unify_bigint\t%l", "write_bigint\t%l", "unify_list", "write_list", "unify_struct\t%f", "write_struct\t%f", "write_unsafe\t%v", "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", "comit_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", "cache_arg\t%r", "cache_sub_arg\t%d", "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", "check_var\t %r", "save_pair\t%v", "save_appl\t%v", "fail_label\t%l", "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%l", "unify_last_longint\t%l", "unify_last_bigint\t%l", "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", "function_to_var\t%v,%B", "function_to_al\t%v,%B", "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 TABLING "table_new_answer", #endif /* TABLING */ #ifdef YAPOR "sync", #endif /* YAPOR */ "fetch_args_for_bccall\t%v", "binary_cfunc\t\t%v,%P", "blob\t%O" #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 () { CELL *OldH = H; cpc = CodeStart; /* MkIntTerm and friends may build terms in the global stack */ H = (CELL *)freep; while (cpc) { ic = cpc->op; arg = cpc->rnd1; rn = cpc->rnd2; cptr = cpc->arnds; if (ic != nop_op) { ShowOp (opformat[ic]); } cpc = cpc->nextInst; } Yap_DebugPutc (Yap_c_error_stream,'\n'); H = OldH; } #endif /* DEBUG */