2011-08-09 12:11:23 +01:00
|
|
|
/*************************************************************************
|
|
|
|
* *
|
|
|
|
* YAP Prolog *
|
|
|
|
* *
|
|
|
|
* Yap Prolog was developed at NCCUP - Universidade do Porto *
|
|
|
|
* *
|
|
|
|
* Copyright L.Damas, V. Santos Costa and Universidade do Porto 1985-- *
|
|
|
|
* *
|
|
|
|
**************************************************************************
|
|
|
|
* *
|
2011-08-24 04:11:54 +01:00
|
|
|
* File: qlyw.c *
|
2011-08-09 12:11:23 +01:00
|
|
|
* comments: quick saver/loader *
|
|
|
|
* *
|
|
|
|
* Last rev: $Date: 2011-08-29$,$Author: vsc $ *
|
|
|
|
* $Log: not supported by cvs2svn $ *
|
|
|
|
* *
|
|
|
|
*************************************************************************/
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
#if DEBUG
|
|
|
|
|
2011-08-24 04:11:54 +01:00
|
|
|
#include <SWI-Stream.h>
|
2011-08-16 14:34:44 +01:00
|
|
|
#include "absmi.h"
|
|
|
|
#include "Foreign.h"
|
|
|
|
#include "alloc.h"
|
|
|
|
#include "yapio.h"
|
|
|
|
#include "iopreds.h"
|
|
|
|
#include "attvar.h"
|
2011-08-09 12:11:23 +01:00
|
|
|
#if HAVE_STRING_H
|
|
|
|
#include <string.h>
|
|
|
|
#endif
|
2011-08-24 04:11:54 +01:00
|
|
|
|
|
|
|
#include "qly.h"
|
2011-08-09 12:11:23 +01:00
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
STATIC_PROTO(void RestoreEntries, (PropEntry *, int USES_REGS));
|
|
|
|
STATIC_PROTO(void CleanCode, (PredEntry * USES_REGS));
|
2011-08-09 12:11:23 +01:00
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static void
|
|
|
|
LookupAtom(Atom at)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-16 14:34:44 +01:00
|
|
|
char *p = RepAtom(at)->StrOfAE;
|
|
|
|
CELL hash = HashFunction((unsigned char *)p) % LOCAL_ExportAtomHashTableSize;
|
|
|
|
export_atom_hash_entry_t *a;
|
2011-08-09 12:11:23 +01:00
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
a = LOCAL_ExportAtomHashChain[hash];
|
|
|
|
while (a) {
|
|
|
|
if (a->val == at) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
a = a->next;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
2011-08-16 14:34:44 +01:00
|
|
|
a = (export_atom_hash_entry_t *)malloc(sizeof(export_atom_hash_entry_t));
|
|
|
|
if (!a) {
|
|
|
|
return;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
2011-08-16 14:34:44 +01:00
|
|
|
a->val = at;
|
|
|
|
a->next = LOCAL_ExportAtomHashChain[hash];
|
|
|
|
LOCAL_ExportAtomHashChain[hash] = a;
|
|
|
|
LOCAL_ExportAtomHashTableNum++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
LookupFunctor(Functor fun)
|
|
|
|
{
|
|
|
|
CELL hash = (CELL)(fun) % LOCAL_ExportFunctorHashTableSize;
|
|
|
|
export_functor_hash_entry_t *f;
|
|
|
|
Atom name = NameOfFunctor(fun);
|
|
|
|
UInt arity = ArityOfFunctor(fun);
|
|
|
|
|
|
|
|
f = LOCAL_ExportFunctorHashChain[hash];
|
|
|
|
while (f) {
|
|
|
|
if (f->name == name && f->arity == arity) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
f = f->next;
|
|
|
|
}
|
|
|
|
f = (export_functor_hash_entry_t *)malloc(sizeof(export_functor_hash_entry_t));
|
|
|
|
if (!f) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
LookupAtom(name);
|
2011-08-24 04:11:54 +01:00
|
|
|
f->val = fun;
|
2011-08-16 14:34:44 +01:00
|
|
|
f->name = name;
|
|
|
|
f->arity = arity;
|
|
|
|
f->next = LOCAL_ExportFunctorHashChain[hash];
|
|
|
|
LOCAL_ExportFunctorHashChain[hash] = f;
|
|
|
|
LOCAL_ExportFunctorHashTableNum++;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
2011-08-24 04:11:54 +01:00
|
|
|
static void
|
|
|
|
LookupPredEntry(PredEntry *pe)
|
|
|
|
{
|
2011-08-24 12:40:06 +01:00
|
|
|
CELL hash = (CELL)(pe) % LOCAL_ExportPredEntryHashTableSize;
|
2011-08-24 04:11:54 +01:00
|
|
|
export_pred_entry_hash_entry_t *p;
|
|
|
|
UInt arity = pe->ArityOfPE;
|
|
|
|
|
|
|
|
p = LOCAL_ExportPredEntryHashChain[hash];
|
|
|
|
while (p) {
|
|
|
|
if (p->val == pe) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
p = p->next;
|
|
|
|
}
|
|
|
|
p = (export_pred_entry_hash_entry_t *)malloc(sizeof(export_pred_entry_hash_entry_t));
|
|
|
|
if (!p) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
p->arity = arity;
|
2011-08-24 12:40:06 +01:00
|
|
|
p->val = pe;
|
2011-08-28 01:23:14 +01:00
|
|
|
if (pe->ModuleOfPred != IDB_MODULE) {
|
|
|
|
if (arity) {
|
|
|
|
p->u.f = pe->FunctorOfPred;
|
|
|
|
LookupFunctor(pe->FunctorOfPred);
|
|
|
|
} else {
|
|
|
|
p->u.a = (Atom)(pe->FunctorOfPred);
|
|
|
|
LookupAtom((Atom)(pe->FunctorOfPred));
|
|
|
|
}
|
2011-08-24 04:11:54 +01:00
|
|
|
} else {
|
2011-08-28 01:23:14 +01:00
|
|
|
if (pe->PredFlags & AtomDBPredFlag) {
|
|
|
|
p->u.a = (Atom)(pe->FunctorOfPred);
|
|
|
|
p->arity = (CELL)(-2);
|
|
|
|
LookupAtom((Atom)(pe->FunctorOfPred));
|
|
|
|
} else if (!(pe->PredFlags & NumberDBPredFlag)) {
|
|
|
|
p->u.f = pe->FunctorOfPred;
|
|
|
|
p->arity = (CELL)(-1);
|
|
|
|
LookupFunctor(pe->FunctorOfPred);
|
|
|
|
} else {
|
|
|
|
p->u.f = pe->FunctorOfPred;
|
|
|
|
}
|
2011-08-24 04:11:54 +01:00
|
|
|
}
|
|
|
|
if (pe->ModuleOfPred) {
|
|
|
|
p->module = AtomOfTerm(pe->ModuleOfPred);
|
|
|
|
} else {
|
|
|
|
p->module = AtomProlog;
|
|
|
|
}
|
2011-08-24 12:40:06 +01:00
|
|
|
LookupAtom(p->module);
|
2011-08-24 04:11:54 +01:00
|
|
|
p->next = LOCAL_ExportPredEntryHashChain[hash];
|
|
|
|
LOCAL_ExportPredEntryHashChain[hash] = p;
|
|
|
|
LOCAL_ExportPredEntryHashTableNum++;
|
|
|
|
}
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static void
|
|
|
|
InitHash(void)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-24 12:40:06 +01:00
|
|
|
LOCAL_ExportFunctorHashTableNum = 0;
|
2011-08-16 14:34:44 +01:00
|
|
|
LOCAL_ExportFunctorHashTableSize = EXPORT_FUNCTOR_TABLE_SIZE;
|
|
|
|
LOCAL_ExportFunctorHashChain = (export_functor_hash_entry_t **)calloc(1, sizeof(export_functor_hash_entry_t *)* LOCAL_ExportFunctorHashTableSize);
|
2011-08-24 12:40:06 +01:00
|
|
|
LOCAL_ExportAtomHashTableNum = 0;
|
2011-08-16 14:34:44 +01:00
|
|
|
LOCAL_ExportAtomHashTableSize = EXPORT_ATOM_TABLE_SIZE;
|
|
|
|
LOCAL_ExportAtomHashChain = (export_atom_hash_entry_t **)calloc(1, sizeof(export_atom_hash_entry_t *)* LOCAL_ExportAtomHashTableSize);
|
2011-08-24 12:40:06 +01:00
|
|
|
LOCAL_ExportPredEntryHashTableNum = 0;
|
2011-08-24 04:11:54 +01:00
|
|
|
LOCAL_ExportPredEntryHashTableSize = EXPORT_PRED_ENTRY_TABLE_SIZE;
|
|
|
|
LOCAL_ExportPredEntryHashChain = (export_pred_entry_hash_entry_t **)calloc(1, sizeof(export_pred_entry_hash_entry_t *)* LOCAL_ExportPredEntryHashTableSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
CloseHash(void)
|
|
|
|
{
|
2011-08-24 12:40:06 +01:00
|
|
|
LOCAL_ExportFunctorHashTableNum = 0;
|
2011-08-24 04:11:54 +01:00
|
|
|
LOCAL_ExportFunctorHashTableSize = 0L;
|
|
|
|
free(LOCAL_ExportFunctorHashChain);
|
2011-08-24 12:40:06 +01:00
|
|
|
LOCAL_ExportAtomHashTableNum = 0;
|
2011-08-24 04:11:54 +01:00
|
|
|
LOCAL_ExportAtomHashTableSize = 0L;
|
|
|
|
free(LOCAL_ExportAtomHashChain);
|
2011-08-24 12:40:06 +01:00
|
|
|
LOCAL_ExportPredEntryHashTableNum = 0;
|
2011-08-24 04:11:54 +01:00
|
|
|
LOCAL_ExportPredEntryHashTableSize = 0L;
|
|
|
|
free(LOCAL_ExportPredEntryHashChain);
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static inline Atom
|
|
|
|
AtomAdjust(Atom a)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-16 14:34:44 +01:00
|
|
|
LookupAtom(a);
|
|
|
|
return a;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static inline Functor
|
|
|
|
FuncAdjust(Functor f)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-16 14:34:44 +01:00
|
|
|
LookupFunctor(f);
|
|
|
|
return f;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static inline Term
|
|
|
|
AtomTermAdjust(Term t)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-16 14:34:44 +01:00
|
|
|
LookupAtom(AtomOfTerm(t));
|
|
|
|
return t;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static inline Term
|
|
|
|
TermToGlobalOrAtomAdjust(Term t)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-16 14:34:44 +01:00
|
|
|
if (t && IsAtomTerm(t))
|
|
|
|
return AtomTermAdjust(t);
|
|
|
|
return t;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
#define IsOldCode(P) FALSE
|
|
|
|
#define IsOldCodeCellPtr(P) FALSE
|
|
|
|
#define IsOldDelay(P) FALSE
|
|
|
|
#define IsOldDelayPtr(P) FALSE
|
|
|
|
#define IsOldLocalInTR(P) FALSE
|
|
|
|
#define IsOldLocalInTRPtr(P) FALSE
|
|
|
|
#define IsOldGlobal(P) FALSE
|
|
|
|
#define IsOldGlobalPtr(P) FALSE
|
|
|
|
#define IsOldTrail(P) FALSE
|
|
|
|
#define IsOldTrailPtr(P) FALSE
|
|
|
|
|
|
|
|
#define CharP(X) ((char *)(X))
|
|
|
|
|
|
|
|
#define REINIT_LOCK(P)
|
|
|
|
#define REINIT_RWLOCK(P)
|
|
|
|
#define BlobTypeAdjust(P) (P)
|
|
|
|
#define NoAGCAtomAdjust(P) (P)
|
|
|
|
#define OrArgAdjust(P)
|
|
|
|
#define TabEntryAdjust(P)
|
|
|
|
#define IntegerAdjust(D) (D)
|
|
|
|
#define AddrAdjust(P) (P)
|
|
|
|
#define MFileAdjust(P) (P)
|
|
|
|
#define CodeVarAdjust(P) (P)
|
|
|
|
#define ConstantAdjust(P) (P)
|
|
|
|
#define ArityAdjust(P) (P)
|
|
|
|
#define DoubleInCodeAdjust(P)
|
|
|
|
#define IntegerInCodeAdjust(P)
|
|
|
|
#define OpcodeAdjust(P) (P)
|
2011-08-24 04:11:54 +01:00
|
|
|
|
|
|
|
static inline Term
|
|
|
|
ModuleAdjust(Term t)
|
|
|
|
{
|
|
|
|
if (!t) return t;
|
|
|
|
return AtomTermAdjust(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline PredEntry *
|
|
|
|
PredEntryAdjust(PredEntry *pe)
|
|
|
|
{
|
|
|
|
LookupPredEntry(pe);
|
|
|
|
return pe;
|
|
|
|
}
|
|
|
|
|
2011-08-24 12:40:06 +01:00
|
|
|
static inline PredEntry *
|
|
|
|
PtoPredAdjust(PredEntry *pe)
|
|
|
|
{
|
|
|
|
LookupPredEntry(pe);
|
|
|
|
return pe;
|
|
|
|
}
|
|
|
|
|
2011-08-24 04:11:54 +01:00
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
#define ExternalFunctionAdjust(P) (P)
|
|
|
|
#define DBRecordAdjust(P) (P)
|
|
|
|
#define PredEntryAdjust(P) (P)
|
|
|
|
#define ModEntryPtrAdjust(P) (P)
|
|
|
|
#define AtomEntryAdjust(P) (P)
|
|
|
|
#define GlobalEntryAdjust(P) (P)
|
|
|
|
#define BlobTermInCodeAdjust(P) (P)
|
|
|
|
#define CellPtoHeapAdjust(P) (P)
|
|
|
|
#define PtoAtomHashEntryAdjust(P) (P)
|
|
|
|
#define CellPtoHeapCellAdjust(P) (P)
|
|
|
|
#define CellPtoTRAdjust(P) (P)
|
|
|
|
#define CodeAddrAdjust(P) (P)
|
|
|
|
#define ConsultObjAdjust(P) (P)
|
|
|
|
#define DelayAddrAdjust(P) (P)
|
|
|
|
#define DelayAdjust(P) (P)
|
|
|
|
#define GlobalAdjust(P) (P)
|
|
|
|
#define DBRefAdjust(P) (P)
|
|
|
|
#define DBRefPAdjust(P) (P)
|
|
|
|
#define DBTermAdjust(P) (P)
|
|
|
|
#define LUIndexAdjust(P) (P)
|
|
|
|
#define SIndexAdjust(P) (P)
|
|
|
|
#define LocalAddrAdjust(P) (P)
|
|
|
|
#define GlobalAddrAdjust(P) (P)
|
|
|
|
#define OpListAdjust(P) (P)
|
|
|
|
#define PtoLUCAdjust(P) (P)
|
|
|
|
#define PtoStCAdjust(P) (P)
|
|
|
|
#define PtoArrayEAdjust(P) (P)
|
|
|
|
#define PtoArraySAdjust(P) (P)
|
|
|
|
#define PtoGlobalEAdjust(P) (P)
|
|
|
|
#define PtoDelayAdjust(P) (P)
|
|
|
|
#define PtoGloAdjust(P) (P)
|
|
|
|
#define PtoLocAdjust(P) (P)
|
|
|
|
#define PtoHeapCellAdjust(P) (P)
|
|
|
|
#define TermToGlobalAdjust(P) (P)
|
|
|
|
#define PtoOpAdjust(P) (P)
|
|
|
|
#define PtoLUClauseAdjust(P) (P)
|
|
|
|
#define PtoLUIndexAdjust(P) (P)
|
|
|
|
#define PtoDBTLAdjust(P) (P)
|
|
|
|
#define PtoPtoPredAdjust(P) (P)
|
|
|
|
#define OpRTableAdjust(P) (P)
|
|
|
|
#define OpEntryAdjust(P) (P)
|
|
|
|
#define PropAdjust(P) (P)
|
|
|
|
#define TrailAddrAdjust(P) (P)
|
|
|
|
#define XAdjust(P) (P)
|
|
|
|
#define YAdjust(P) (P)
|
|
|
|
#define HoldEntryAdjust(P) (P)
|
|
|
|
#define CodeCharPAdjust(P) (P)
|
|
|
|
#define CodeVoidPAdjust(P) (P)
|
|
|
|
#define HaltHookAdjust(P) (P)
|
|
|
|
|
|
|
|
#define recompute_mask(dbr)
|
|
|
|
|
|
|
|
#define rehash(oldcode, NOfE, KindOfEntries)
|
|
|
|
|
|
|
|
#define RestoreSWIHash()
|
|
|
|
|
|
|
|
#include "rheap.h"
|
|
|
|
|
|
|
|
static void
|
|
|
|
RestoreHashPreds( USES_REGS1 )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
RestoreAtomList(Atom atm USES_REGS)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t save_bytes(IOSTREAM *stream, void *ptr, size_t sz)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-16 14:34:44 +01:00
|
|
|
return Sfwrite(ptr, sz, 1, stream);
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static size_t save_byte(IOSTREAM *stream, int byte)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-24 04:11:54 +01:00
|
|
|
Sputc(byte, stream);
|
|
|
|
return 1;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
2011-08-28 01:23:14 +01:00
|
|
|
static size_t save_bits16(IOSTREAM *stream, BITS16 val)
|
|
|
|
{
|
|
|
|
BITS16 v = val;
|
|
|
|
return save_bytes(stream, &v, sizeof(BITS16));
|
|
|
|
}
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static size_t save_uint(IOSTREAM *stream, UInt val)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-16 14:34:44 +01:00
|
|
|
UInt v = val;
|
|
|
|
return save_bytes(stream, &v, sizeof(UInt));
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
2011-08-24 04:11:54 +01:00
|
|
|
static size_t save_int(IOSTREAM *stream, int val)
|
|
|
|
{
|
|
|
|
UInt v = val;
|
|
|
|
return save_bytes(stream, &v, sizeof(int));
|
|
|
|
}
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static size_t save_tag(IOSTREAM *stream, qlf_tag_t tag)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
2011-08-16 14:34:44 +01:00
|
|
|
return save_byte(stream, tag);
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static int
|
|
|
|
SaveHash(IOSTREAM *stream)
|
2011-08-09 12:11:23 +01:00
|
|
|
{
|
|
|
|
UInt i;
|
2011-08-24 04:11:54 +01:00
|
|
|
/* first, current opcodes */
|
|
|
|
CHECK(save_tag(stream, QLY_START_X));
|
|
|
|
save_uint(stream, (UInt)&ARG1);
|
|
|
|
CHECK(save_tag(stream, QLY_START_OPCODES));
|
|
|
|
save_int(stream, _std_top);
|
|
|
|
for (i= 0; i < _std_top; i++) {
|
|
|
|
save_uint(stream, (UInt)Yap_opcode(i));
|
|
|
|
}
|
|
|
|
CHECK(save_tag(stream, QLY_START_ATOMS));
|
2011-08-16 14:34:44 +01:00
|
|
|
CHECK(save_uint(stream, LOCAL_ExportAtomHashTableNum));
|
|
|
|
for (i = 0; i < LOCAL_ExportAtomHashTableSize; i++) {
|
|
|
|
export_atom_hash_entry_t *a = LOCAL_ExportAtomHashChain[i];
|
|
|
|
while (a) {
|
|
|
|
export_atom_hash_entry_t *a0 = a;
|
|
|
|
Atom at = a->val;
|
|
|
|
CHECK(save_uint(stream, (UInt)at));
|
|
|
|
if (IsWideAtom(at)) {
|
|
|
|
CHECK(save_tag(stream, QLY_ATOM_WIDE));
|
|
|
|
CHECK(save_uint(stream, wcslen(RepAtom(at)->WStrOfAE)));
|
2011-08-24 04:11:54 +01:00
|
|
|
CHECK(save_bytes(stream, at->WStrOfAE, (wcslen(at->WStrOfAE)+1)*sizeof(wchar_t)));
|
2011-08-16 14:34:44 +01:00
|
|
|
} else {
|
|
|
|
CHECK(save_tag(stream, QLY_ATOM));
|
|
|
|
CHECK(save_uint(stream, strlen(RepAtom(at)->StrOfAE)));
|
|
|
|
CHECK(save_bytes(stream, at->StrOfAE, (strlen(at->StrOfAE)+1)*sizeof(char)));
|
|
|
|
}
|
|
|
|
a = a->next;
|
|
|
|
free(a0);
|
|
|
|
}
|
|
|
|
}
|
2011-08-24 04:11:54 +01:00
|
|
|
save_tag(stream, QLY_START_FUNCTORS);
|
2011-08-16 14:34:44 +01:00
|
|
|
save_uint(stream, LOCAL_ExportFunctorHashTableNum);
|
|
|
|
for (i = 0; i < LOCAL_ExportFunctorHashTableSize; i++) {
|
|
|
|
export_functor_hash_entry_t *f = LOCAL_ExportFunctorHashChain[i];
|
|
|
|
while (f) {
|
|
|
|
export_functor_hash_entry_t *f0 = f;
|
2011-08-24 04:11:54 +01:00
|
|
|
CHECK(save_uint(stream, (UInt)(f->val)));
|
2011-08-16 14:34:44 +01:00
|
|
|
CHECK(save_uint(stream, f->arity));
|
|
|
|
CHECK(save_uint(stream, (CELL)(f->name)));
|
|
|
|
f = f->next;
|
|
|
|
free(f0);
|
|
|
|
}
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
2011-08-24 04:11:54 +01:00
|
|
|
save_tag(stream, QLY_START_PRED_ENTRIES);
|
|
|
|
save_uint(stream, LOCAL_ExportPredEntryHashTableNum);
|
|
|
|
for (i = 0; i < LOCAL_ExportPredEntryHashTableSize; i++) {
|
|
|
|
export_pred_entry_hash_entry_t *p = LOCAL_ExportPredEntryHashChain[i];
|
|
|
|
while (p) {
|
|
|
|
export_pred_entry_hash_entry_t *p0 = p;
|
|
|
|
CHECK(save_uint(stream, (UInt)(p->val)));
|
|
|
|
CHECK(save_uint(stream, p->arity));
|
|
|
|
CHECK(save_uint(stream, (UInt)p->module));
|
|
|
|
CHECK(save_uint(stream, (UInt)p->u.f));
|
|
|
|
p = p->next;
|
|
|
|
free(p0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static size_t
|
|
|
|
save_clauses(IOSTREAM *stream, PredEntry *pp) {
|
|
|
|
yamop *FirstC, *LastC;
|
|
|
|
|
|
|
|
FirstC = pp->cs.p_code.FirstClause;
|
|
|
|
LastC = pp->cs.p_code.LastClause;
|
|
|
|
if (FirstC == NULL && LastC == NULL) {
|
2011-08-24 04:11:54 +01:00
|
|
|
return 1;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
if (pp->PredFlags & LogUpdatePredFlag) {
|
|
|
|
LogUpdClause *cl = ClauseCodeToLogUpdClause(FirstC);
|
|
|
|
|
|
|
|
while (cl != NULL) {
|
2011-08-28 01:23:14 +01:00
|
|
|
if (pp->TimeStampOfPred >= cl->ClTimeStart &&
|
|
|
|
pp->TimeStampOfPred <= cl->ClTimeEnd) {
|
|
|
|
UInt size = cl->ClSize;
|
|
|
|
CHECK(save_uint(stream, (UInt)cl));
|
|
|
|
CHECK(save_uint(stream, size));
|
|
|
|
CHECK(save_bytes(stream, cl, size));
|
|
|
|
}
|
2011-08-09 12:11:23 +01:00
|
|
|
cl = cl->ClNext;
|
|
|
|
}
|
|
|
|
} else if (pp->PredFlags & MegaClausePredFlag) {
|
|
|
|
MegaClause *cl = ClauseCodeToMegaClause(FirstC);
|
2011-08-16 14:34:44 +01:00
|
|
|
UInt size = cl->ClSize;
|
2011-08-09 12:11:23 +01:00
|
|
|
|
2011-08-24 04:11:54 +01:00
|
|
|
CHECK(save_uint(stream, (UInt)cl));
|
|
|
|
CHECK(save_uint(stream, size));
|
2011-08-16 14:34:44 +01:00
|
|
|
CHECK(save_bytes(stream, cl, size));
|
2011-08-09 12:11:23 +01:00
|
|
|
} else if (pp->PredFlags & DynamicPredFlag) {
|
|
|
|
yamop *cl = FirstC;
|
|
|
|
|
|
|
|
do {
|
2011-08-16 14:34:44 +01:00
|
|
|
DynamicClause *dcl = ClauseCodeToDynamicClause(cl);
|
|
|
|
UInt size = dcl->ClSize;
|
|
|
|
|
2011-08-24 04:11:54 +01:00
|
|
|
CHECK(save_uint(stream, (UInt)cl));
|
|
|
|
CHECK(save_uint(stream, size));
|
2011-08-16 14:34:44 +01:00
|
|
|
CHECK(save_bytes(stream, dcl, size));
|
2011-08-09 12:11:23 +01:00
|
|
|
if (cl == LastC) return 1;
|
|
|
|
cl = NextDynamicClause(cl);
|
|
|
|
} while (TRUE);
|
|
|
|
} else {
|
|
|
|
StaticClause *cl = ClauseCodeToStaticClause(FirstC);
|
|
|
|
|
|
|
|
do {
|
2011-08-16 14:34:44 +01:00
|
|
|
UInt size = cl->ClSize;
|
|
|
|
|
2011-08-24 04:11:54 +01:00
|
|
|
CHECK(save_uint(stream, (UInt)cl));
|
|
|
|
CHECK(save_uint(stream, size));
|
2011-08-16 14:34:44 +01:00
|
|
|
CHECK(save_bytes(stream, cl, size));
|
2011-08-09 12:11:23 +01:00
|
|
|
if (cl->ClCode == LastC) return 1;
|
|
|
|
cl = cl->ClNext;
|
|
|
|
} while (TRUE);
|
|
|
|
}
|
2011-08-24 04:11:54 +01:00
|
|
|
return 1;
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static size_t
|
|
|
|
save_pred(IOSTREAM *stream, PredEntry *ap) {
|
|
|
|
CHECK(save_uint(stream, ap->ArityOfPE));
|
2011-08-24 04:11:54 +01:00
|
|
|
CHECK(save_uint(stream, (UInt)(ap->FunctorOfPred)));
|
2011-08-09 12:11:23 +01:00
|
|
|
CHECK(save_uint(stream, ap->PredFlags));
|
|
|
|
CHECK(save_uint(stream, ap->cs.p_code.NOfClauses));
|
2011-08-28 01:23:14 +01:00
|
|
|
CHECK(save_uint(stream, ap->src.IndxId));
|
2011-08-09 12:11:23 +01:00
|
|
|
return save_clauses(stream, ap);
|
|
|
|
}
|
|
|
|
|
2011-08-16 14:34:44 +01:00
|
|
|
static int
|
|
|
|
clean_pred(PredEntry *pp USES_REGS) {
|
|
|
|
if (pp->PredFlags & (AsmPredFlag|CPredFlag)) {
|
|
|
|
/* assembly */
|
|
|
|
if (pp->CodeOfPred) {
|
|
|
|
CleanClauses(pp->CodeOfPred, pp->CodeOfPred, pp PASS_REGS);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
CleanClauses(pp->cs.p_code.FirstClause, pp->cs.p_code.LastClause, pp PASS_REGS);
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2011-08-28 01:23:14 +01:00
|
|
|
static size_t
|
|
|
|
mark_pred(PredEntry *ap)
|
|
|
|
{
|
|
|
|
if (ap->ModuleOfPred != IDB_MODULE) {
|
|
|
|
if (ap->ArityOfPE) {
|
|
|
|
FuncAdjust(ap->FunctorOfPred);
|
|
|
|
} else {
|
|
|
|
AtomAdjust((Atom)(ap->FunctorOfPred));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (ap->PredFlags & AtomDBPredFlag) {
|
|
|
|
AtomAdjust((Atom)(ap->FunctorOfPred));
|
|
|
|
} else if (!(ap->PredFlags & NumberDBPredFlag)) {
|
|
|
|
FuncAdjust(ap->FunctorOfPred);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!(ap->PredFlags & (MultiFileFlag|NumberDBPredFlag)) &&
|
|
|
|
ap->src.OwnerFile) {
|
|
|
|
AtomAdjust(ap->src.OwnerFile);
|
|
|
|
}
|
|
|
|
CHECK(clean_pred(ap PASS_REGS));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t
|
|
|
|
mark_ops(IOSTREAM *stream, Term mod) {
|
|
|
|
OpEntry *op = OpList;
|
|
|
|
while (op) {
|
|
|
|
if (!mod || op->OpModule == mod) {
|
|
|
|
AtomAdjust(op->OpName);
|
|
|
|
if (op->OpModule)
|
|
|
|
AtomTermAdjust(op->OpModule);
|
|
|
|
}
|
|
|
|
op = op->OpNext;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t
|
|
|
|
save_ops(IOSTREAM *stream, Term mod) {
|
|
|
|
OpEntry *op = OpList;
|
|
|
|
while (op) {
|
|
|
|
if (!mod || op->OpModule == mod) {
|
|
|
|
CHECK(save_tag(stream, QLY_NEW_OP));
|
|
|
|
save_uint(stream, (UInt)op->OpName);
|
|
|
|
save_uint(stream, (UInt)op->OpModule);
|
|
|
|
save_bits16(stream, op->Prefix);
|
|
|
|
save_bits16(stream, op->Infix);
|
|
|
|
save_bits16(stream, op->Posfix);
|
|
|
|
}
|
|
|
|
op = op->OpNext;
|
|
|
|
}
|
|
|
|
CHECK(save_tag(stream, QLY_END_OPS));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2011-08-09 12:11:23 +01:00
|
|
|
static size_t
|
|
|
|
save_module(IOSTREAM *stream, Term mod) {
|
2011-08-16 14:34:44 +01:00
|
|
|
CACHE_REGS
|
2011-08-09 12:11:23 +01:00
|
|
|
PredEntry *ap = Yap_ModulePred(mod);
|
2011-08-16 14:34:44 +01:00
|
|
|
InitHash();
|
2011-08-24 12:40:06 +01:00
|
|
|
ModuleAdjust(mod);
|
2011-08-16 14:34:44 +01:00
|
|
|
while (ap) {
|
2011-08-28 01:23:14 +01:00
|
|
|
CHECK(mark_pred(ap));
|
2011-08-16 14:34:44 +01:00
|
|
|
ap = ap->NextPredOfModule;
|
|
|
|
}
|
2011-08-28 01:23:14 +01:00
|
|
|
/* just to make sure */
|
|
|
|
mark_ops(stream, mod);
|
2011-08-16 14:34:44 +01:00
|
|
|
SaveHash(stream);
|
2011-08-24 04:11:54 +01:00
|
|
|
CHECK(save_tag(stream, QLY_START_MODULE));
|
|
|
|
CHECK(save_uint(stream, (UInt)mod));
|
2011-08-16 14:34:44 +01:00
|
|
|
ap = Yap_ModulePred(mod);
|
2011-08-09 12:11:23 +01:00
|
|
|
while (ap) {
|
2011-08-24 04:11:54 +01:00
|
|
|
CHECK(save_tag(stream, QLY_START_PREDICATE));
|
2011-08-09 12:11:23 +01:00
|
|
|
CHECK(save_pred(stream, ap));
|
|
|
|
ap = ap->NextPredOfModule;
|
|
|
|
}
|
2011-08-24 04:11:54 +01:00
|
|
|
CHECK(save_tag(stream, QLY_END_PREDICATES));
|
2011-08-28 01:23:14 +01:00
|
|
|
CHECK(save_tag(stream, QLY_END_MODULES));
|
|
|
|
save_ops(stream, mod);
|
|
|
|
CloseHash();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t
|
|
|
|
save_program(IOSTREAM *stream) {
|
|
|
|
CACHE_REGS
|
|
|
|
ModEntry *me = CurrentModules;
|
|
|
|
|
|
|
|
InitHash();
|
|
|
|
/* should we allow the user to see hidden predicates? */
|
|
|
|
while (me) {
|
|
|
|
PredEntry *pp;
|
|
|
|
AtomAdjust(me->AtomOfME);
|
|
|
|
pp = me->PredForME;
|
|
|
|
while (pp != NULL) {
|
|
|
|
CHECK(mark_pred(pp));
|
|
|
|
pp = pp->NextPredOfModule;
|
|
|
|
}
|
|
|
|
me = me->NextME;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* just to make sure */
|
|
|
|
mark_ops(stream, 0);
|
|
|
|
SaveHash(stream);
|
|
|
|
me = CurrentModules;
|
|
|
|
while (me) {
|
|
|
|
PredEntry *pp;
|
|
|
|
pp = me->PredForME;
|
|
|
|
CHECK(save_tag(stream, QLY_START_MODULE));
|
|
|
|
CHECK(save_uint(stream, (UInt)MkAtomTerm(me->AtomOfME)));
|
|
|
|
while (pp != NULL) {
|
|
|
|
CHECK(save_tag(stream, QLY_START_PREDICATE));
|
|
|
|
CHECK(save_pred(stream, pp));
|
|
|
|
pp = pp->NextPredOfModule;
|
|
|
|
}
|
|
|
|
CHECK(save_tag(stream, QLY_END_PREDICATES));
|
|
|
|
me = me->NextME;
|
|
|
|
}
|
|
|
|
CHECK(save_tag(stream, QLY_END_MODULES));
|
|
|
|
save_ops(stream, 0);
|
2011-08-24 04:11:54 +01:00
|
|
|
CloseHash();
|
2011-08-09 12:11:23 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Int
|
|
|
|
p_save_module_preds( USES_REGS1 )
|
|
|
|
{
|
|
|
|
IOSTREAM *stream;
|
|
|
|
Term tmod = Deref(ARG2);
|
|
|
|
|
|
|
|
if (!Yap_getOutputStream(Yap_InitSlot(Deref(ARG1) PASS_REGS), &stream)) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (IsVarTerm(tmod)) {
|
|
|
|
Yap_Error(INSTANTIATION_ERROR,tmod,"save_module/2");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (!IsAtomTerm(tmod)) {
|
|
|
|
Yap_Error(TYPE_ERROR_ATOM,tmod,"save_module/2");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return save_module(stream, tmod) != 0;
|
|
|
|
}
|
|
|
|
|
2011-08-28 01:23:14 +01:00
|
|
|
static Int
|
|
|
|
p_save_program( USES_REGS1 )
|
|
|
|
{
|
|
|
|
IOSTREAM *stream;
|
|
|
|
|
|
|
|
if (!Yap_getOutputStream(Yap_InitSlot(Deref(ARG1) PASS_REGS), &stream)) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return save_program(stream) != 0;
|
|
|
|
}
|
|
|
|
|
2011-08-09 12:34:23 +01:00
|
|
|
#endif
|
|
|
|
|
2011-08-09 12:11:23 +01:00
|
|
|
void Yap_InitQLY(void)
|
|
|
|
{
|
2011-08-09 12:34:23 +01:00
|
|
|
#if DEBUG
|
2011-08-28 01:23:14 +01:00
|
|
|
Yap_InitCPred("$qsave_module_preds", 2, p_save_module_preds, SyncPredFlag|HiddenPredFlag|UserCPredFlag);
|
|
|
|
Yap_InitCPred("$qsave_program", 1, p_save_program, SyncPredFlag|HiddenPredFlag|UserCPredFlag);
|
2011-08-09 12:34:23 +01:00
|
|
|
#endif
|
2011-08-09 12:11:23 +01:00
|
|
|
}
|
|
|
|
|