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/save.c

1654 lines
42 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 *
* *
**************************************************************************
* *
2017-11-11 02:15:59 +00:00
* File: save.c * Last
*rev: * mods:
** comments: saving and restoring a Prolog computation *
* *
*************************************************************************/
#ifdef SCCS
2017-11-11 02:15:59 +00:00
static char SccsId[] = "@(#)save.c 1.3 3/15/90";
#endif
2016-01-31 10:18:12 +00:00
#include "absmi.h"
#include "alloc.h"
#if _MSC_VER || defined(__MINGW32__)
2013-11-16 16:58:07 +00:00
#if HAVE_WINSOCK2_H
#include <winsock2.h>
#endif
#include <psapi.h>
2017-11-11 02:15:59 +00:00
#include <windows.h>
#endif
#if USE_DL_MALLOC
#include "dlmalloc.h"
#endif
2017-11-11 02:15:59 +00:00
#include "Foreign.h"
2015-10-20 08:06:46 +01:00
#include "YapText.h"
#include "sshift.h"
2016-01-31 10:18:12 +00:00
#include "yapio.h"
#if HAVE_STRING_H
#include <string.h>
#endif
#if !HAVE_STRNCAT
2017-11-11 02:15:59 +00:00
#define strncat(X, Y, Z) strcat(X, Y)
#endif
#if !HAVE_STRNCPY
2017-11-11 02:15:59 +00:00
#define strncpy(X, Y, Z) strcpy(X, Y)
#endif
#if HAVE_FCNTL_H
#include <fcntl.h>
#endif
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#ifdef HAVE_SYS_TYPES_H
#include <sys/types.h>
#endif
#ifdef HAVE_SYS_STAT_H
#include <sys/stat.h>
#endif
#include "iopreds.h"
/********* hack for accesing several kinds of terms. Should be cleaned **/
2017-11-11 02:15:59 +00:00
static char end_msg[256] = "*** End of YAP saved state *****";
2011-02-15 19:24:15 +00:00
/* SWI IO, must be restarted after restore */
void initIO(void);
#ifdef DEBUG
/*
*
#FOR DEBUGGING define DEBUG_RESTORE0 to check the file stuff,
#define DEBUG_RESTORE1 to see if it is able to prepare the chain,
#define DEBUG_RESTORE2 to see how things are going,
#define DEBUG_RESTORE3 to check if the atom chain is still a working chain,
* define DEBUG_RESTORE4 if you want to set the output for some
* particular file,
* define DEBUG_RESTORE5 if you want to see how the stacks are being
* cleaned up,
* define DEBUG_RESTORE6 if you want to follow the execution in
*
* Also a file is defined where you can write things, by default stderr
*
* Good Luck
*/
#endif
2017-11-11 02:15:59 +00:00
static int myread(FILE *, char *, Int);
static Int mywrite(FILE *, char *, Int);
2015-06-19 01:30:13 +01:00
static FILE *open_file(char *, int);
2017-11-11 02:15:59 +00:00
static int close_file(void);
static Int putout(CELL);
static Int putcellptr(CELL *);
static CELL get_cell(void);
static CELL *get_cellptr(/* CELL * */ void);
static int put_info(int, int CACHE_TYPE);
static int save_regs(int CACHE_TYPE);
static int save_code_info(void);
static int save_heap(void);
static int save_stacks(int CACHE_TYPE);
static int save_crc(void);
static Int do_save(int CACHE_TYPE);
static Int p_save2(CACHE_TYPE1);
static Int p_save_program(CACHE_TYPE1);
static int check_header(CELL *, CELL *, CELL *, CELL *CACHE_TYPE);
static int get_heap_info(CACHE_TYPE1);
static int get_regs(int CACHE_TYPE);
static int get_insts(OPCODE[]);
static int get_hash(void);
static int CopyCode(CACHE_TYPE1);
static int CopyStacks(CACHE_TYPE1);
static int get_coded(int, OPCODE[] CACHE_TYPE);
static void restore_codes(void);
static void RestoreDB(DBEntry *CACHE_TYPE);
static void RestoreDBTerm(DBTerm *, bool, int CACHE_TYPE);
static void CleanClauses(yamop *First, yamop *Last, PredEntry *pp USES_REGS);
static void rehash(CELL *, int, int CACHE_TYPE);
static void CleanCode(PredEntry *CACHE_TYPE);
static void RestoreEntries(PropEntry *, int CACHE_TYPE);
static void RestoreFreeSpace(CACHE_TYPE1);
static void restore_heap(void);
#ifdef DEBUG_RESTORE3
2017-11-11 02:15:59 +00:00
static void ShowAtoms(void);
static void ShowEntries(PropEntry *);
#endif
2017-11-11 02:15:59 +00:00
static int OpenRestore(const char *, const char *, CELL *, CELL *, CELL *,
CELL *, FILE **);
static void CloseRestore(void);
#ifndef _WIN32
2017-11-11 02:15:59 +00:00
static int check_opcodes(OPCODE[]);
#endif
2017-11-11 02:15:59 +00:00
static void RestoreHeap(OPCODE[] CACHE_TYPE);
static Int p_restore(CACHE_TYPE1);
static void restore_heap_regs(CACHE_TYPE1);
static void restore_regs(int CACHE_TYPE);
#ifdef MACYAP
2013-04-25 23:15:04 +01:00
static void NewFileInfo(long, long);
2017-11-11 02:15:59 +00:00
extern int DefVol;
#endif
#ifdef _WIN32
#if HAVE_IO_H
#include <io.h>
#endif
#endif
#ifdef LIGHT
#include <strings.h>
2017-11-11 02:15:59 +00:00
#include <unix.h>
2017-11-11 02:15:59 +00:00
void LightBug(char *);
2017-11-11 02:15:59 +00:00
static void LightBug(s) char *s;
{}
2017-11-11 02:15:59 +00:00
#endif /* LIGHT */
2017-11-11 02:15:59 +00:00
static Int do_SYSTEM_ERROR_INTERNAL(yap_error_number etype, const char *msg) {
CACHE_REGS
2017-10-27 23:06:23 +01:00
char *buf = malloc(1043);
#if HAVE_SNPRINTF
#if HAVE_STRERROR
2017-11-11 02:15:59 +00:00
snprintf(buf, 1043 - 1, "%s (%s when reading %s)", msg, strerror(errno),
LOCAL_FileNameBuf);
#else
2017-11-11 02:15:59 +00:00
snprintf(buf, 1024 - 1, "%s, (system error %d when reading %s)", msg, errno,
LOCAL_FileNameBuf);
#endif
#else
#if HAVE_STRERROR
2017-11-11 02:15:59 +00:00
snprintf(buf, 1024 - 1, "%s, (%s when reading %s)", msg, strerror(errno),
LOCAL_FileNameBuf);
#else
2017-11-11 02:15:59 +00:00
snprintf(buf, 1024 - 1, "%s, (system error %d when reading %s)", msg, errno,
LOCAL_FileNameBuf);
#endif
#endif
LOCAL_Error_TYPE = etype;
return -1;
}
2017-11-11 02:15:59 +00:00
inline static int myread(FILE *fd, char *buffer, Int len) {
size_t nread;
while (len > 0) {
2017-11-11 02:15:59 +00:00
nread = fread(buffer, 1, (int)len, fd);
if (nread < 1) {
2017-11-11 02:15:59 +00:00
return do_SYSTEM_ERROR_INTERNAL(PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM,
"bad read on saved state");
}
buffer += nread;
len -= nread;
}
return len;
}
2017-11-11 02:15:59 +00:00
inline static Int mywrite(FILE *fd, char *buff, Int len) {
size_t nwritten;
2017-10-27 23:06:23 +01:00
while (len > 0) {
2015-06-19 01:30:13 +01:00
nwritten = fwrite(buff, 1, (size_t)len, fd);
if ((long int)nwritten < 0) {
2017-11-11 02:15:59 +00:00
return do_SYSTEM_ERROR_INTERNAL(SYSTEM_ERROR_INTERNAL,
"bad write on saved state");
}
buff += nwritten;
len -= nwritten;
}
2010-03-22 16:56:58 +00:00
return len;
}
2017-11-11 02:15:59 +00:00
#define FullSaved 1
/* Where the code was before */
2017-11-11 02:15:59 +00:00
typedef CELL *CELLPOINTER;
2015-06-19 01:30:13 +01:00
static FILE *splfild = NULL;
#ifdef DEBUG
#ifdef DEBUG_RESTORE4
2017-11-11 02:15:59 +00:00
static FILE *errout;
#else
#define errout GLOBAL_stderr
#endif
2017-11-11 02:15:59 +00:00
#endif /* DEBUG */
2017-11-11 02:15:59 +00:00
static Int OldHeapUsed;
2017-11-11 02:15:59 +00:00
static CELL which_save;
/* Open a file to read or to write */
2017-11-11 02:15:59 +00:00
static FILE *open_file(char *my_file, int flag) {
2015-06-19 01:30:13 +01:00
FILE *splfild;
char flags[6];
2017-11-11 02:15:59 +00:00
int i = 0;
if (flag & O_RDONLY) {
flags[i++] = 'r';
}
if (flag & O_CREAT) {
flags[i++] = 'w';
}
if (flag & O_WRONLY) {
flags[i++] = 'w';
}
if (flag & O_APPEND) {
flags[i++] = 'a';
}
#if _WIN32
if (flag & O_BINARY) {
flags[i++] = 'b';
}
#endif
flags[i] = '\0';
2017-11-11 02:15:59 +00:00
splfild = fopen(my_file, flags);
#ifdef undf0
fprintf(errout, "Opened file %s\n", my_file);
#endif
2014-11-09 11:16:24 +00:00
return splfild;
}
2017-11-11 02:15:59 +00:00
static int close_file(void) {
if (splfild == 0)
2010-03-22 16:56:58 +00:00
return 0;
2015-06-19 01:30:13 +01:00
if (fclose(splfild) < 0)
2017-11-11 02:15:59 +00:00
return do_SYSTEM_ERROR_INTERNAL(SYSTEM_ERROR_INTERNAL,
"bad close on saved state");
splfild = 0;
2010-03-22 16:56:58 +00:00
return 1;
}
/* stores a cell in a file */
2017-11-11 02:15:59 +00:00
static Int putout(CELL l) { return mywrite(splfild, (char *)&l, sizeof(CELL)); }
/* stores a pointer to a cell in a file */
2017-11-11 02:15:59 +00:00
static Int putcellptr(CELL *l) {
return mywrite(splfild, (char *)&l, sizeof(CELLPOINTER));
}
/* gets a cell from a file */
2017-11-11 02:15:59 +00:00
static CELL get_cell(void) {
CELL l;
myread(splfild, (char *)&l, Unsigned(sizeof(CELL)));
return (l);
}
/* gets a cell from a file */
2017-11-11 02:15:59 +00:00
static CELL get_header_cell(void) {
CELL l;
2014-10-25 16:46:13 +01:00
size_t count = 0;
int n;
while (count < sizeof(CELL)) {
2017-11-11 02:15:59 +00:00
if ((n = fread(&l, 1, sizeof(CELL) - count, splfild)) < 0) {
do_SYSTEM_ERROR_INTERNAL(PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM,
"failed to read saved state header");
return 0L;
}
count += n;
}
return l;
}
/* gets a pointer to cell from a file */
2017-11-11 02:15:59 +00:00
static CELL *get_cellptr(void) {
CELL *l;
2017-11-11 02:15:59 +00:00
if (myread(splfild, (char *)&l, Unsigned(sizeof(CELLPOINTER))) < 0)
2010-03-22 16:56:58 +00:00
return NULL;
return (l);
}
/*
* writes the header (at the moment YAPV*), info about what kind of saved
* set, the work size, and the space ocuppied
*/
2017-11-11 02:15:59 +00:00
static int put_info(int info, int mode USES_REGS) {
char msg[256 * 16];
2017-11-11 02:15:59 +00:00
sprintf(msg,
"#!/bin/sh\nexec_dir=${YAPBINDIR:-%s}\nexec $exec_dir/yap $0 "
"\"$@\"\n%cYAP-%s",
YAP_BINDIR, 1, YAP_FULL_VERSION);
2010-03-22 16:56:58 +00:00
if (mywrite(splfild, msg, strlen(msg) + 1))
return -1;
if (putout(Unsigned(info)) < 0)
return -1;
/* say whether we just saved the heap or everything */
2010-03-22 16:56:58 +00:00
if (putout(mode) < 0)
return -1;
/* current state of stacks, to be used by SavedInfo */
/* space available in heap area */
2017-11-11 02:15:59 +00:00
if (putout(Unsigned(LOCAL_GlobalBase) - Unsigned(Yap_HeapBase)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
/* space available for stacks */
2017-11-11 02:15:59 +00:00
if (putout(Unsigned(LOCAL_LocalBase) - Unsigned(LOCAL_GlobalBase)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
/* space available for trail */
2017-11-11 02:15:59 +00:00
if (putout(Unsigned(LOCAL_TrailTop) - Unsigned(LOCAL_TrailBase)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
/* Space used in heap area */
2017-11-11 02:15:59 +00:00
if (putout(Unsigned(HeapTop) - Unsigned(Yap_HeapBase)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
/* Space used for local stack */
2017-11-11 02:15:59 +00:00
if (putout(Unsigned(LCL0) - Unsigned(ASP)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
/* Space used for global stack */
2014-01-19 21:15:05 +00:00
if (putout(Unsigned(HR) - Unsigned(LOCAL_GlobalBase)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
/* Space used for trail */
if (putout(Unsigned(TR) - Unsigned(LOCAL_TrailBase)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
return 0;
}
2017-11-11 02:15:59 +00:00
static int save_regs(int mode USES_REGS) {
/* save all registers */
2010-03-22 16:56:58 +00:00
if (putout((CELL)compile_arrays) < 0)
return -1;
if (mode == DO_EVERYTHING) {
2010-03-22 16:56:58 +00:00
if (putcellptr((CELL *)CP) < 0)
return -1;
if (putcellptr(ENV) < 0)
return -1;
if (putcellptr(ASP) < 0)
return -1;
/* if (putout((CELL)N) < 0)
return -1; */
if (putcellptr(H0) < 0)
return -1;
if (putcellptr(LCL0) < 0)
return -1;
2014-01-19 21:15:05 +00:00
if (putcellptr(HR) < 0)
2010-03-22 16:56:58 +00:00
return -1;
if (putcellptr(HB) < 0)
return -1;
if (putcellptr((CELL *)B) < 0)
return -1;
if (putcellptr((CELL *)TR) < 0)
return -1;
if (putcellptr(YENV) < 0)
return -1;
if (putcellptr(S) < 0)
return -1;
if (putcellptr((CELL *)P) < 0)
return -1;
if (putout(CreepFlag) < 0)
return -1;
2013-12-12 14:24:40 +00:00
if (putout(EventFlag) < 0)
return -1;
2011-03-30 15:32:59 +01:00
#if defined(YAPOR_SBA) || defined(TABLING)
2010-03-22 16:56:58 +00:00
if (putcellptr(H_FZ) < 0)
return -1;
if (putcellptr((CELL *)B_FZ) < 0)
return -1;
if (putcellptr((CELL *)TR_FZ) < 0)
return -1;
2011-03-30 15:32:59 +01:00
#endif /* YAPOR_SBA || TABLING */
}
2010-03-22 16:56:58 +00:00
if (putout(CurrentModule) < 0)
return -1;
if (mode == DO_EVERYTHING) {
#ifdef COROUTINING
2011-05-04 10:11:41 +01:00
if (putout(LOCAL_WokenGoals) < 0)
2010-03-22 16:56:58 +00:00
return -1;
#endif
2017-11-11 02:15:59 +00:00
#ifdef DEPTH_LIMIT
2010-03-22 16:56:58 +00:00
if (putout(DEPTH) < 0)
return -1;
#endif
2011-05-04 10:11:41 +01:00
if (putout(LOCAL_GcGeneration) < 0)
2010-03-22 16:56:58 +00:00
return -1;
2011-05-04 10:11:41 +01:00
if (putout(LOCAL_GcPhase) < 0)
2010-03-22 16:56:58 +00:00
return -1;
2011-05-04 10:11:41 +01:00
if (putout(LOCAL_GcCurrentPhase) < 0)
2010-03-22 16:56:58 +00:00
return -1;
}
/* The operand base */
2010-03-22 16:56:58 +00:00
if (putcellptr(CellPtr(XREGS)) < 0)
return -1;
if (putout(which_save) < 0)
return -1;
/* Now start by saving the code */
/* the heap boundaries */
2010-03-22 16:56:58 +00:00
if (putcellptr(CellPtr(Yap_HeapBase)) < 0)
return -1;
if (putcellptr(CellPtr(HeapTop)) < 0)
return -1;
/* and the space it ocuppies */
2016-01-03 02:06:09 +00:00
if (putout(Unsigned(HeapUsed)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
/* Then the start of the free code */
2010-03-22 16:56:58 +00:00
if (putcellptr(CellPtr(FreeBlocks)) < 0)
return -1;
if (putcellptr(CellPtr(AuxBase)) < 0)
return -1;
2010-03-22 16:56:58 +00:00
if (putcellptr(AuxSp) < 0)
return -1;
if (putcellptr(CellPtr(AuxTop)) < 0)
return -1;
2011-05-04 10:11:41 +01:00
if (putcellptr(CellPtr(LOCAL_ScratchPad.ptr)) < 0)
return -1;
2011-05-04 10:11:41 +01:00
if (putout(LOCAL_ScratchPad.sz) < 0)
return -1;
2011-05-04 10:11:41 +01:00
if (putout(LOCAL_ScratchPad.msz) < 0)
return -1;
if (mode == DO_EVERYTHING) {
/* put the old trail base, just in case it moves again */
2010-03-22 16:56:58 +00:00
if (putout(ARG1) < 0)
return -1;
if (which_save == 2) {
2010-03-22 16:56:58 +00:00
if (putout(ARG2) < 0)
2017-11-11 02:15:59 +00:00
return -1;
}
if (putcellptr(CellPtr(LOCAL_TrailBase)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
}
2010-03-22 16:56:58 +00:00
return 0;
}
2017-11-11 02:15:59 +00:00
static int save_code_info(void) {
/* First the instructions */
{
op_numbers i;
2017-11-11 02:15:59 +00:00
OPCODE my_ops[_std_top + 1];
for (i = _Ystop; i <= _std_top; ++i)
my_ops[i] = Yap_opcode(i);
2017-11-11 02:15:59 +00:00
if (mywrite(splfild, (char *)my_ops, sizeof(OPCODE) * (_std_top + 1)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
}
/* and the current character codes */
2017-11-11 02:15:59 +00:00
if (mywrite(splfild, (char *)Yap_chtype,
NUMBER_OF_CHARS * sizeof(char_kind_t)) < 0)
2010-03-22 16:56:58 +00:00
return -1;
return 0;
}
2017-11-11 02:15:59 +00:00
static int save_heap(void) {
#ifdef USE_SYSTEM_MALLOC
2010-03-30 09:16:07 +01:00
return -1;
#endif
int j;
/* Then save the whole heap */
Yap_ResetConsultStack();
j = Unsigned(HeapTop) - Unsigned(Yap_HeapBase);
/* store 10 more cells because of the memory manager */
2017-11-11 02:15:59 +00:00
if (mywrite(splfild, (char *)Yap_HeapBase, j) < 0)
2010-03-22 16:56:58 +00:00
return -1;
return 0;
}
2017-11-11 02:15:59 +00:00
static int save_stacks(int mode USES_REGS) {
int j;
switch (mode) {
case DO_EVERYTHING:
/* Now, go and save the state */
/* Save the local stack */
j = Unsigned(LCL0) - Unsigned(ASP);
2017-11-11 02:15:59 +00:00
if (mywrite(splfild, (char *)ASP, j) < 0)
2010-03-22 16:56:58 +00:00
return -1;
/* Save the global stack */
2014-01-19 21:15:05 +00:00
j = Unsigned(HR) - Unsigned(LOCAL_GlobalBase);
2017-11-11 02:15:59 +00:00
if (mywrite(splfild, (char *)LOCAL_GlobalBase, j) < 0)
2010-03-22 16:56:58 +00:00
return -1;
/* Save the trail */
j = Unsigned(TR) - Unsigned(LOCAL_TrailBase);
2017-11-11 02:15:59 +00:00
if (mywrite(splfild, (char *)LOCAL_TrailBase, j) < 0)
2010-03-22 16:56:58 +00:00
return -1;
break;
2017-11-11 02:15:59 +00:00
case DO_ONLY_CODE: {
tr_fr_ptr tr_ptr = TR;
while (tr_ptr != (tr_fr_ptr)LOCAL_TrailBase) {
CELL val = TrailTerm(tr_ptr - 1);
if (IsVarTerm(val)) {
CELL *d1 = VarOfTerm(val);
if (d1 < (CELL *)HeapTop) {
if (putout(val) < 0)
return -1;
}
} else if (IsPairTerm(val)) {
CELL *d1 = RepPair(val);
if (d1 < (CELL *)HeapTop) {
if (putout(val) < 0)
return -1;
}
}
2017-11-11 02:15:59 +00:00
tr_ptr--;
}
2017-11-11 02:15:59 +00:00
}
2010-03-22 16:56:58 +00:00
if (putcellptr(NULL) < 0)
return -1;
break;
}
2010-03-22 16:56:58 +00:00
return 0;
}
2017-11-11 02:15:59 +00:00
static int save_crc(void) {
/* Save a CRC */
2010-03-22 16:56:58 +00:00
return mywrite(splfild, end_msg, 256);
}
2017-11-11 02:15:59 +00:00
static Int do_save(int mode USES_REGS) {
Term t1 = Deref(ARG1);
if (Yap_HoleSize) {
2017-11-11 02:15:59 +00:00
Yap_Error(SYSTEM_ERROR_INTERNAL,
MkAtomTerm(Yap_LookupAtom(LOCAL_FileNameBuf)),
"restore/1: address space has holes of size %ld, cannot save",
(long int)Yap_HoleSize);
return FALSE;
}
if (!Yap_GetName(LOCAL_FileNameBuf, YAP_FILENAME_MAX, t1)) {
2017-11-11 02:15:59 +00:00
Yap_Error(TYPE_ERROR_LIST, t1, "save/1");
return FALSE;
}
Yap_CloseStreams(TRUE);
if ((splfild = open_file(LOCAL_FileNameBuf, O_WRONLY | O_CREAT)) < 0) {
2017-11-11 02:15:59 +00:00
Yap_Error(SYSTEM_ERROR_INTERNAL,
MkAtomTerm(Yap_LookupAtom(LOCAL_FileNameBuf)),
"restore/1, open(%s)", strerror(errno));
return (FALSE);
}
if (put_info(FullSaved, mode PASS_REGS) < 0)
2010-03-22 16:56:58 +00:00
return -1;
if (save_regs(mode PASS_REGS) < 0)
2010-03-22 16:56:58 +00:00
return -1;
if (save_code_info() < 0)
return -1;
if (save_heap() < 0)
return -1;
if (save_stacks(mode PASS_REGS) < 0)
2010-03-22 16:56:58 +00:00
return -1;
if (save_crc() < 0)
return -1;
close_file();
return (TRUE);
}
/* Saves a complete prolog environment */
2017-11-11 02:15:59 +00:00
static Int p_save2(USES_REGS1) {
Int res;
yhandle_t CurSlot = Yap_StartSlots();
2010-11-25 16:33:25 +00:00
Term t;
2011-04-29 14:59:17 +01:00
#ifdef YAPOR
if (GLOBAL_number_workers != 1) {
2017-11-11 02:15:59 +00:00
Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
"cannot perform save: more than a worker/thread running");
return (FALSE);
}
2011-04-29 14:59:17 +01:00
#endif /* YAPOR */
#ifdef THREADS
if (GLOBAL_NOfThreads != 1) {
2017-11-11 02:15:59 +00:00
Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
"cannot perform save: more than a worker/thread running");
return (FALSE);
}
2011-04-29 14:59:17 +01:00
#endif /* THREADS */
/* avoid double saves */
if (IsNonVarTerm(t = Deref(ARG2)))
return TRUE;
2017-11-11 02:15:59 +00:00
if (!Yap_unify(ARG2, MkIntTerm(1)))
return FALSE;
which_save = 2;
2015-02-03 02:36:51 +00:00
CurSlot = Yap_StartSlots();
res = do_save(DO_EVERYTHING PASS_REGS);
2015-02-03 02:36:51 +00:00
Yap_CloseSlots(CurSlot);
return res;
}
/* Just save the program, not the stacks */
2017-11-11 02:15:59 +00:00
static Int p_save_program(USES_REGS1) {
which_save = 0;
return do_save(DO_ONLY_CODE PASS_REGS);
}
/* Now, to restore the saved code */
/* First check out if we are dealing with a valid file */
2017-11-11 02:15:59 +00:00
static int check_header(CELL *info, CELL *ATrail, CELL *AStack,
CELL *AHeap USES_REGS) {
char pp[256];
char msg[256];
CELL hp_size, gb_size, lc_size, tr_size, mode;
int n;
/* make sure we always check if there are enough bytes */
/* skip the first line */
pp[0] = '\0';
do {
2015-06-19 01:30:13 +01:00
if ((n = fread(pp, 1, 1, splfild)) <= 0) {
2017-11-11 02:15:59 +00:00
do_SYSTEM_ERROR_INTERNAL(PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM,
"failed to scan first line from saved state");
return FAIL_RESTORE;
}
} while (pp[0] != 1);
/* now check the version */
2014-02-16 12:57:00 +00:00
sprintf(msg, "YAP-%s", YAP_FULL_VERSION);
{
int count = 0, n, to_read = Unsigned(strlen(msg) + 1);
while (count < to_read) {
2017-11-11 02:15:59 +00:00
if ((n = fread(pp, 1, to_read - count, splfild)) <= 0) {
do_SYSTEM_ERROR_INTERNAL(
PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM,
"failed to scan version info from saved state");
return FAIL_RESTORE;
}
count += n;
}
}
if (strcmp(pp, msg) != 0) {
2017-11-11 02:15:59 +00:00
strncpy(LOCAL_ErrorMessage, "saved state ", MAX_ERROR_MSG_SIZE - 1);
strncat(LOCAL_ErrorMessage, LOCAL_FileNameBuf, MAX_ERROR_MSG_SIZE - 1);
strncat(LOCAL_ErrorMessage, " failed to match version ID",
MAX_ERROR_MSG_SIZE - 1);
2015-09-25 10:57:26 +01:00
LOCAL_Error_TYPE = SYSTEM_ERROR_SAVED_STATE;
return FAIL_RESTORE;
}
/* check info on header */
/* ignore info on saved state */
*info = get_header_cell();
if (LOCAL_ErrorMessage)
return FAIL_RESTORE;
/* check the restore mode */
mode = get_header_cell();
if (LOCAL_ErrorMessage)
return FAIL_RESTORE;
if (mode != DO_EVERYTHING && mode != DO_ONLY_CODE) {
return FAIL_RESTORE;
}
/* ignore info on stacks size */
*AHeap = get_header_cell();
if (LOCAL_ErrorMessage) {
return FAIL_RESTORE;
}
*AStack = get_header_cell();
if (LOCAL_ErrorMessage) {
return FAIL_RESTORE;
}
*ATrail = get_header_cell();
if (LOCAL_ErrorMessage) {
return FAIL_RESTORE;
}
/* now, check whether we got enough enough space to load the
saved space */
hp_size = get_cell();
if (LOCAL_ErrorMessage)
return FAIL_RESTORE;
while (Yap_HeapBase != NULL &&
2017-11-11 02:15:59 +00:00
hp_size > Unsigned(HeapLim) - Unsigned(Yap_HeapBase)) {
if (!Yap_growheap(FALSE, hp_size, NULL)) {
return FAIL_RESTORE;
}
}
if (mode == DO_EVERYTHING) {
lc_size = get_cell();
if (LOCAL_ErrorMessage)
return FAIL_RESTORE;
2017-11-11 02:15:59 +00:00
gb_size = get_cell();
if (LOCAL_ErrorMessage)
return FAIL_RESTORE;
2017-11-11 02:15:59 +00:00
if (Yap_HeapBase != NULL &&
lc_size + gb_size >
Unsigned(LOCAL_LocalBase) - Unsigned(LOCAL_GlobalBase)) {
if (LOCAL_ErrorMessage != NULL)
2017-11-11 02:15:59 +00:00
LOCAL_ErrorMessage = "could not allocate enough stack space";
return FAIL_RESTORE;
}
2017-11-11 02:15:59 +00:00
if (Yap_HeapBase != NULL &&
(tr_size = get_cell()) >
Unsigned(LOCAL_TrailTop) - Unsigned(LOCAL_TrailBase)) {
if (LOCAL_ErrorMessage != NULL)
2017-11-11 02:15:59 +00:00
LOCAL_ErrorMessage = "could not allocate enough trail space";
return FAIL_RESTORE;
}
} else {
/* skip cell size */
get_header_cell();
if (LOCAL_ErrorMessage)
return FAIL_RESTORE;
get_header_cell();
if (LOCAL_ErrorMessage)
return FAIL_RESTORE;
get_header_cell();
if (LOCAL_ErrorMessage)
return FAIL_RESTORE;
}
2017-11-11 02:15:59 +00:00
return (mode);
}
/* Gets the state of the heap, and evaluates the related variables */
2017-11-11 02:15:59 +00:00
static int get_heap_info(USES_REGS1) {
LOCAL_OldHeapBase = (ADDR)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
2017-11-11 02:15:59 +00:00
LOCAL_OldHeapTop = (ADDR)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
2017-11-11 02:15:59 +00:00
OldHeapUsed = (Int)get_cell();
if (LOCAL_ErrorMessage)
return -1;
2017-11-11 02:15:59 +00:00
FreeBlocks = (BlockHeader *)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
AuxBase = (ADDR)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
AuxSp = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
AuxTop = (ADDR)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
2011-05-04 10:11:41 +01:00
LOCAL_ScratchPad.ptr = (ADDR)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
2011-05-04 10:11:41 +01:00
LOCAL_ScratchPad.sz = get_cell();
if (LOCAL_ErrorMessage)
return -1;
2011-05-04 10:11:41 +01:00
LOCAL_ScratchPad.msz = get_cell();
if (LOCAL_ErrorMessage)
return -1;
2011-05-04 10:11:41 +01:00
LOCAL_HDiff = Unsigned(Yap_HeapBase) - Unsigned(LOCAL_OldHeapBase);
return 1;
}
/* Gets the register array */
/* Saves the old bases for the work areas */
/* and evaluates the difference from the old areas to the new ones */
2017-11-11 02:15:59 +00:00
static int get_regs(int flag USES_REGS) {
CELL *NewGlobalBase = (CELL *)LOCAL_GlobalBase;
CELL *NewLCL0 = LCL0;
CELL *OldXREGS;
/* Get regs */
compile_arrays = (int)get_cell();
if (LOCAL_ErrorMessage)
return -1;
if (flag == DO_EVERYTHING) {
CP = (yamop *)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
ENV = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
ASP = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
/* N = get_cell(); */
H0 = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
LCL0 = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
2014-01-19 21:15:05 +00:00
HR = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
HB = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
B = (choiceptr)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
TR = (tr_fr_ptr)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
YENV = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
S = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
P = (yamop *)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
CreepFlag = get_cell();
2013-12-12 14:24:40 +00:00
if (LOCAL_ErrorMessage)
return -1;
EventFlag = get_cell();
if (LOCAL_ErrorMessage)
return -1;
2011-03-30 15:32:59 +01:00
#if defined(YAPOR_SBA) || defined(TABLING)
H_FZ = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
B_FZ = (choiceptr)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
TR_FZ = (tr_fr_ptr)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
2011-03-30 15:32:59 +01:00
#endif /* YAPOR_SBA || TABLING */
}
CurrentModule = get_cell();
if (LOCAL_ErrorMessage)
return -1;
if (flag == DO_EVERYTHING) {
#ifdef COROUTINING
2011-05-04 10:11:41 +01:00
LOCAL_WokenGoals = get_cell();
if (LOCAL_ErrorMessage)
return -1;
#endif
2017-11-11 02:15:59 +00:00
#ifdef DEPTH_LIMIT
DEPTH = get_cell();
if (LOCAL_ErrorMessage)
return -1;
#endif
2011-05-04 10:11:41 +01:00
LOCAL_GcGeneration = get_cell();
if (LOCAL_ErrorMessage)
return -1;
2011-05-04 10:11:41 +01:00
LOCAL_GcPhase = get_cell();
if (LOCAL_ErrorMessage)
return -1;
2011-05-04 10:11:41 +01:00
LOCAL_GcCurrentPhase = get_cell();
if (LOCAL_ErrorMessage)
return -1;
}
/* Get the old bases */
OldXREGS = get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
which_save = get_cell();
if (LOCAL_ErrorMessage)
return -1;
2017-11-11 02:15:59 +00:00
LOCAL_XDiff = (CELL)XREGS - (CELL)OldXREGS;
if (LOCAL_ErrorMessage)
return -1;
2017-11-11 02:15:59 +00:00
if (get_heap_info(PASS_REGS1) < 0)
return -1;
if (flag == DO_EVERYTHING) {
ARG1 = get_cell();
if (LOCAL_ErrorMessage)
return -1;
if (which_save == 2) {
ARG2 = get_cell();
if (LOCAL_ErrorMessage)
2017-11-11 02:15:59 +00:00
return -1;
}
/* get old trail base */
2011-05-04 10:11:41 +01:00
LOCAL_OldTrailBase = (ADDR)get_cellptr();
if (LOCAL_ErrorMessage)
return -1;
/* Save the old register where we can easily access them */
2011-05-04 10:11:41 +01:00
LOCAL_OldASP = ASP;
LOCAL_OldLCL0 = LCL0;
LOCAL_OldGlobalBase = (CELL *)LOCAL_GlobalBase;
2014-01-19 21:15:05 +00:00
LOCAL_OldH = HR;
2011-05-04 10:11:41 +01:00
LOCAL_OldTR = TR;
LOCAL_GDiff = Unsigned(NewGlobalBase) - Unsigned(LOCAL_GlobalBase);
2011-05-04 10:11:41 +01:00
LOCAL_GDiff0 = 0;
LOCAL_LDiff = Unsigned(NewLCL0) - Unsigned(LCL0);
LOCAL_TrDiff = LOCAL_LDiff;
LOCAL_GlobalBase = (ADDR)NewGlobalBase;
LCL0 = NewLCL0;
}
return 1;
}
/* Get the old opcodes and place them in a hash table */
2017-11-11 02:15:59 +00:00
static int get_insts(OPCODE old_ops[]) {
return myread(splfild, (char *)old_ops, sizeof(OPCODE) * (_std_top + 1));
}
/* Get the old atoms hash table */
2017-11-11 02:15:59 +00:00
static int get_hash(void) {
return myread(splfild, (char *)Yap_chtype,
NUMBER_OF_CHARS * sizeof(char_kind_t));
}
/* Copy all of the old code to the new Heap */
2017-11-11 02:15:59 +00:00
static int CopyCode(USES_REGS1) {
if (myread(splfild, (char *)Yap_HeapBase,
(Unsigned(LOCAL_OldHeapTop) - Unsigned(LOCAL_OldHeapBase))) < 0) {
return -1;
}
return 1;
}
/* Copy the local and global stack and also the trail to their new home */
/* In REGS we still have nonadjusted values !! */
2017-11-11 02:15:59 +00:00
static int CopyStacks(USES_REGS1) {
Int j;
char *NewASP;
2011-05-04 10:11:41 +01:00
j = Unsigned(LOCAL_OldLCL0) - Unsigned(ASP);
2017-11-11 02:15:59 +00:00
NewASP = (char *)(Unsigned(ASP) + (Unsigned(LCL0) - Unsigned(LOCAL_OldLCL0)));
if (myread(splfild, (char *)NewASP, j) < 0)
return -1;
2014-01-19 21:15:05 +00:00
j = Unsigned(HR) - Unsigned(LOCAL_OldGlobalBase);
2017-11-11 02:15:59 +00:00
if (myread(splfild, (char *)LOCAL_GlobalBase, j) < 0)
return -1;
2011-05-04 10:11:41 +01:00
j = Unsigned(TR) - Unsigned(LOCAL_OldTrailBase);
if (myread(splfild, LOCAL_TrailBase, j))
return -1;
return 1;
}
/* Copy the local and global stack and also the trail to their new home */
/* In REGS we still have nonadjusted values !! */
2017-11-11 02:15:59 +00:00
static int CopyTrailEntries(USES_REGS1) {
CELL entry, *Entries;
Entries = (CELL *)LOCAL_TrailBase;
do {
*Entries++ = entry = get_cell();
if (LOCAL_ErrorMessage)
return -1;
} while ((CODEADDR)entry != NULL);
return 1;
}
/* get things which are saved in the file */
2017-11-11 02:15:59 +00:00
static int get_coded(int flag, OPCODE old_ops[] USES_REGS) {
char my_end_msg[256];
if (get_regs(flag PASS_REGS) < 0)
return -1;
if (get_insts(old_ops) < 0)
return -1;
if (get_hash() < 0)
return -1;
2017-11-11 02:15:59 +00:00
if (CopyCode(PASS_REGS1) < 0)
return -1;
switch (flag) {
case DO_EVERYTHING:
2017-11-11 02:15:59 +00:00
if (CopyStacks(PASS_REGS1) < 0)
return -1;
break;
case DO_ONLY_CODE:
2017-11-11 02:15:59 +00:00
if (CopyTrailEntries(PASS_REGS1) < 0)
return -1;
break;
}
/* Check CRC */
if (myread(splfild, my_end_msg, 256) < 0)
return -1;
2017-11-11 02:15:59 +00:00
if (strcmp(end_msg, my_end_msg) != 0) {
LOCAL_ErrorMessage = "bad trailing CRC in saved state";
return -1;
}
return 1;
}
/* restore some heap registers */
2017-11-11 02:15:59 +00:00
static void restore_heap_regs(USES_REGS1) {
if (HeapTop) {
HeapTop = AddrAdjust(HeapTop);
2017-11-11 02:15:59 +00:00
*((YAP_SEG_SIZE *)HeapTop) = InUseFlag;
}
2016-07-31 10:35:00 +01:00
// HeapMax = HeapUsed = OldHeapUsed;
HeapLim = LOCAL_GlobalBase;
}
/* adjust abstract machine registers */
2017-11-11 02:15:59 +00:00
static void restore_regs(int flag USES_REGS) {
restore_heap_regs(PASS_REGS1);
if (CurrentModule) {
2017-11-11 02:15:59 +00:00
CurrentModule = AtomTermAdjust(CurrentModule);
;
}
if (flag == DO_EVERYTHING) {
CP = PtoOpAdjust(CP);
ENV = PtoLocAdjust(ENV);
ASP = PtoLocAdjust(ASP);
2014-01-19 21:15:05 +00:00
HR = PtoGloAdjust(HR);
B = (choiceptr)PtoLocAdjust(CellPtr(B));
TR = PtoTRAdjust(TR);
P = PtoOpAdjust(P);
HB = PtoLocAdjust(HB);
YENV = PtoLocAdjust(YENV);
S = PtoGloAdjust(S);
2011-05-04 10:11:41 +01:00
LOCAL_WokenGoals = AbsAppl(PtoGloAdjust(RepAppl(LOCAL_WokenGoals)));
}
}
2017-11-11 02:15:59 +00:00
static void recompute_mask(DBRef dbr) {
if (dbr->Flags & DBNoVars) {
2017-11-11 02:15:59 +00:00
dbr->Mask = Yap_EvalMasks((Term)dbr->DBT.Entry, &(dbr->Key));
} else if (dbr->Flags & DBComplex) {
/* This is quite nasty, we want to recalculate the mask but
we don't want to rebuild the whole term. We'll just build whatever we
need to recompute the mask.
*/
CELL *x = (CELL *)HeapTop, *tp;
unsigned int Arity, i;
Term out;
2017-11-11 02:15:59 +00:00
char *tbase = CharP(dbr->DBT.Contents - 1);
if (IsPairTerm(dbr->DBT.Entry)) {
out = AbsPair(x);
Arity = 2;
2017-11-11 02:15:59 +00:00
tp = (CELL *)(tbase + (CELL)RepPair(dbr->DBT.Entry));
} else {
Functor f;
2017-11-11 02:15:59 +00:00
tp = (CELL *)(tbase + (CELL)RepAppl(dbr->DBT.Entry));
f = (Functor)(*tp++);
out = AbsAppl(x);
Arity = ArityOfFunctor(f);
*x++ = (CELL)f;
2017-11-11 02:15:59 +00:00
if (Arity > 3)
Arity = 3;
}
for (i = 0; i < Arity; i++) {
2017-11-11 02:15:59 +00:00
register Term tw = *tp++;
if (IsVarTerm(tw)) {
2017-11-11 02:15:59 +00:00
RESET_VARIABLE(x);
} else if (IsApplTerm(tw)) {
2017-11-11 02:15:59 +00:00
/* just fetch the functor from where it is in the data-base.
This guarantees we have access to references and friends. */
CELL offset = (CELL)RepAppl(tw);
if (offset > dbr->DBT.NOfCells * sizeof(CELL))
*x = tw;
else
*x = AbsAppl((CELL *)(tbase + offset));
} else if (IsAtomicTerm(tw)) {
2017-11-11 02:15:59 +00:00
*x = tw;
} else if (IsPairTerm(tw)) {
2017-11-11 02:15:59 +00:00
*x = AbsPair(x);
}
x++;
}
dbr->Mask = Yap_EvalMasks(out, &(dbr->Key));
}
}
#define HASH_SHIFT 6
/*
* This is used to make an hash table correct, after displacing its elements,
* HCEnd should point to an area of free space, usually in the heap. The
* routine is very dependent on the hash function used, and it destroys the
* previous "hit" order
*/
2017-11-11 02:15:59 +00:00
static void rehash(CELL *oldcode, int NOfE, int KindOfEntries USES_REGS) {
register CELL *savep, *basep;
CELL *oldp = oldcode;
int TableSize = NOfE - 1, NOfEntries;
register int i;
int hash;
CELL WorkTerm, failplace = 0;
CELL *Base = oldcode;
2011-05-04 10:11:41 +01:00
if (LOCAL_HDiff == 0)
return;
2014-01-19 21:15:05 +00:00
basep = HR;
2017-11-11 02:15:59 +00:00
if (HR + (NOfE * 2) > ASP) {
basep = (CELL *)TR;
2017-11-11 02:15:59 +00:00
if (basep + (NOfE * 2) > (CELL *)LOCAL_TrailTop) {
if (!Yap_growtrail((ADDR)(basep + (NOfE * 2)) - LOCAL_TrailTop, TRUE)) {
Yap_Error(RESOURCE_ERROR_TRAIL, TermNil,
"not enough space to restore hash tables for indexing");
Yap_exit(1);
}
}
}
for (i = 0; i < NOfE; ++i) {
if (*oldp == 0) {
failplace = oldp[1];
break;
}
oldp += 2;
}
savep = basep;
oldp = oldcode;
for (i = 0; i < NOfE; ++i) {
if (*oldp != 0) {
savep[0] = oldp[0];
savep[1] = oldp[1];
oldp[0] = 0;
oldp[1] = failplace;
savep += 2;
}
oldp += 2;
}
2017-11-11 02:15:59 +00:00
NOfEntries = (savep - basep) / 2;
savep = basep;
for (i = 0; i < NOfEntries; ++i) {
2017-11-11 02:15:59 +00:00
register Int d;
CELL *hentry;
2017-11-11 02:15:59 +00:00
WorkTerm = savep[i * 2];
hash = (Unsigned(WorkTerm) >> HASH_SHIFT) & TableSize;
hentry = Base + hash * 2;
d = TableSize & (Unsigned(WorkTerm) | 1);
while (*hentry) {
#ifdef DEBUG
#ifdef CLASHES
++clashes;
#endif /* CLASHES */
#endif /* DEBUG */
hash = (hash + d) & TableSize;
hentry = Base + hash * 2;
}
hentry[0] = WorkTerm;
2017-11-11 02:15:59 +00:00
hentry[1] = savep[i * 2 + 1];
}
}
2017-11-11 02:15:59 +00:00
static void RestoreFlags(UInt NFlags) {}
2015-08-07 22:57:53 +01:00
#include "rheap.h"
/* restore the atom entries which are invisible for the user */
2017-11-11 02:15:59 +00:00
static void RestoreIOStructures(void) { Yap_InitStdStreams(); }
2017-11-11 02:15:59 +00:00
static void RestoreFreeSpace(USES_REGS1) {
#if USE_DL_MALLOC
Yap_av = (struct malloc_state *)AddrAdjust((ADDR)Yap_av);
Yap_RestoreDLMalloc();
if (AuxSp != NULL) {
2011-05-04 10:11:41 +01:00
if (AuxBase < LOCAL_OldHeapBase || AuxBase > LOCAL_OldHeapTop) {
AuxSp = NULL;
AuxBase = NULL;
AuxTop = NULL;
} else {
AuxSp = PtoHeapCellAdjust(AuxSp);
AuxBase = AddrAdjust(AuxBase);
AuxTop = AddrAdjust(AuxTop);
2011-05-04 10:11:41 +01:00
LOCAL_ScratchPad.ptr = AddrAdjust(LOCAL_ScratchPad.ptr);
}
}
#else
/* restores the list of free space, with its curious structure */
BlockHeader *bpt, *bsz;
if (FreeBlocks != NULL)
FreeBlocks = BlockAdjust(FreeBlocks);
bpt = FreeBlocks;
if (AuxSp != NULL)
AuxSp = CellPtoHeapAdjust(AuxSp);
if (AuxTop != NULL)
AuxTop = AddrAdjust(AuxTop);
while (bpt != NULL) {
if (bpt->b_next != NULL) {
bsz = bpt->b_next = BlockAdjust(bpt->b_next);
while (bsz != NULL) {
if (bsz->b_next_size != NULL)
2017-11-11 02:15:59 +00:00
bsz->b_next_size = BlockAdjust(bsz->b_next_size);
if (bsz->b_next != NULL)
2017-11-11 02:15:59 +00:00
bsz->b_next = BlockAdjust(bsz->b_next);
bsz = bsz->b_next;
}
}
if (bpt->b_next_size != NULL)
bpt->b_next_size = BlockAdjust(bpt->b_next_size);
bpt = bpt->b_next_size;
}
2017-11-11 02:15:59 +00:00
*((YAP_SEG_SIZE *)HeapTop) = InUseFlag;
#endif
}
2017-11-11 02:15:59 +00:00
static void RestoreAtomList(Atom atm USES_REGS) {
AtomEntry *at;
at = RepAtom(atm);
if (EndOfPAEntr(at))
return;
do {
RestoreAtom(at PASS_REGS);
at = RepAtom(at->NextOfAE);
} while (!EndOfPAEntr(at));
}
2017-11-11 02:15:59 +00:00
static void RestoreHashPreds(USES_REGS1) {
UInt size = PredHashTableSize;
2014-09-10 05:50:43 +01:00
bool malloced = FALSE;
PredEntry **np;
UInt i;
PredEntry **oldp = PredHash;
2017-11-11 02:15:59 +00:00
np = (PredEntry **)Yap_AllocAtomSpace(sizeof(PredEntry *) * size);
if (!np) {
2017-11-11 02:15:59 +00:00
if (!(np = (PredEntry **)malloc(sizeof(PredEntry *) * size))) {
Yap_Error(SYSTEM_ERROR_FATAL, TermNil,
"Could not allocate space for pred table");
return;
}
malloced = TRUE;
}
for (i = 0; i < size; i++) {
np[i] = NULL;
}
for (i = 0; i < PredHashTableSize; i++) {
PredEntry *p = oldp[i];
if (p)
p = PredEntryAdjust(p);
while (p) {
Prop nextp;
UInt hsh;
if (p->NextOfPE)
2017-11-11 02:15:59 +00:00
p->NextOfPE = PropAdjust(p->NextOfPE);
nextp = p->NextOfPE;
CleanCode(p PASS_REGS);
hsh = PRED_HASH(p->FunctorOfPred, p->ModuleOfPred, size);
p->NextOfPE = AbsPredProp(np[hsh]);
np[hsh] = p;
p = RepPredProp(nextp);
}
}
for (i = 0; i < size; i++) {
PredHash[i] = np[i];
}
if (malloced)
free((ADDR)np);
else
Yap_FreeAtomSpace((ADDR)np);
}
/*
* This is the really tough part, to restore the whole of the heap
*/
2017-11-11 02:15:59 +00:00
static void restore_heap(void) {
2009-10-30 23:59:00 +00:00
restore_codes();
RestoreIOStructures();
}
#ifdef DEBUG_RESTORE3
2017-11-11 02:15:59 +00:00
static void ShowEntries(pp) PropEntry *pp;
{
while (!EndOfPAEntr(pp)) {
2017-11-11 02:15:59 +00:00
fprintf(GLOBAL_stderr, "Estou a ver a prop %x em %x\n", pp->KindOfPE, pp);
pp = RepProp(pp->NextOfPE);
}
}
2017-11-11 02:15:59 +00:00
static void ShowAtoms() {
AtomHashEntry *HashPtr = HashChain;
register int i;
for (i = 0; i < AtomHashTableSize; ++i) {
if (HashPtr->Entry != NIL) {
2017-11-11 02:15:59 +00:00
AtomEntry *at;
at = RepAtom(HashPtr->Entry);
do {
2017-11-11 02:15:59 +00:00
fprintf(GLOBAL_stderr, "Passei ao %s em %x\n", at->StrOfAE, at);
ShowEntries(RepProp(at->PropsOfAE));
} while (!EndOfPAEntr(at = RepAtom(at->NextOfAE)));
}
HashPtr++;
}
HashPtr = WideHashChain;
for (i = 0; i < WideAtomHashTableSize; ++i) {
if (HashPtr->Entry != NIL) {
2017-11-11 02:15:59 +00:00
AtomEntry *at;
at = RepAtom(HashPtr->Entry);
do {
2017-11-11 02:15:59 +00:00
fprintf(GLOBAL_stderr, "Passei ao %s em %x\n", at->StrOfAE, at);
ShowEntries(RepProp(at->PropsOfAE));
} while (!EndOfPAEntr(at = RepAtom(at->NextOfAE)));
}
HashPtr++;
}
}
#endif /* DEBUG_RESTORE3 */
#include <stdio.h>
2017-11-11 02:15:59 +00:00
static int commit_to_saved_state(char *s, CELL *Astate, CELL *ATrail,
CELL *AStack, CELL *AHeap) {
2017-10-27 23:06:23 +01:00
CACHE_REGS
2017-11-11 02:15:59 +00:00
int mode;
2017-11-11 02:15:59 +00:00
if ((mode = check_header(Astate, ATrail, AStack, AHeap PASS_REGS)) ==
FAIL_RESTORE)
return (FAIL_RESTORE);
LOCAL_PrologMode = BootMode;
if (Yap_HeapBase) {
2017-11-11 02:15:59 +00:00
if (falseGlobalPrologFlag(HALT_AFTER_CONSULT_FLAG) && !silentMode()) {
Yap_findFile(s, NULL, NULL, LOCAL_FileNameBuf2, true, YAP_QLY, true,
true);
fprintf(stderr, "%% Restoring file %s\n", LOCAL_FileNameBuf2);
}
Yap_CloseStreams(TRUE);
}
#ifdef DEBUG_RESTORE4
/*
* This should be another file, like the log file
*/
errout = GLOBAL_stderr;
#endif
2010-08-17 13:19:17 +01:00
return mode;
}
2017-11-11 02:15:59 +00:00
static int try_open(char *inpf, CELL *Astate, CELL *ATrail, CELL *AStack,
CELL *AHeap, FILE **streamp) {
2010-08-17 13:19:17 +01:00
int mode;
2011-08-31 21:59:30 +01:00
if (streamp) {
2015-06-19 01:30:13 +01:00
if ((*streamp = fopen(inpf, "rb"))) {
2011-09-01 14:11:38 +01:00
return DO_ONLY_CODE;
}
return FAIL_RESTORE;
2011-08-31 21:59:30 +01:00
}
2010-08-17 13:19:17 +01:00
if ((splfild = open_file(inpf, O_RDONLY)) < 0) {
return FAIL_RESTORE;
}
2017-11-11 02:15:59 +00:00
if ((mode = commit_to_saved_state(inpf, Astate, ATrail, AStack, AHeap)) !=
FAIL_RESTORE) {
CACHE_REGS
2017-11-11 02:15:59 +00:00
LOCAL_ErrorMessage = NULL;
2010-08-17 13:19:17 +01:00
return mode;
}
return mode;
}
2017-11-11 02:15:59 +00:00
static int OpenRestore(const char *inpf, const char *YapLibDir, CELL *Astate,
CELL *ATrail, CELL *AStack, CELL *AHeap,
FILE **streamp) {
CACHE_REGS
2017-10-27 23:06:23 +01:00
int mode;
2017-11-11 02:15:59 +00:00
char fname[YAP_FILENAME_MAX + 1];
2016-02-13 02:56:19 +00:00
2017-11-11 02:15:59 +00:00
if (!Yap_findFile(inpf, YAP_STARTUP, YapLibDir, fname, true, YAP_SAVED_STATE,
true, true))
2017-06-18 11:14:55 +01:00
return FAIL_RESTORE;
2017-11-11 02:15:59 +00:00
if (fname[0] && (mode = try_open(fname, Astate, ATrail, AStack, AHeap,
streamp)) != FAIL_RESTORE) {
setAtomicGlobalPrologFlag(RESOURCE_DATABASE_FLAG,
MkAtomTerm(Yap_LookupAtom(fname)));
return mode;
}
/* try to open from current directory */
/* could not open file */
if (LOCAL_ErrorMessage == NULL) {
2017-11-11 02:15:59 +00:00
do_SYSTEM_ERROR_INTERNAL(PERMISSION_ERROR_OPEN_SOURCE_SINK,
"incorrect saved state ");
} else {
2017-11-11 02:15:59 +00:00
strncpy(LOCAL_FileNameBuf, inpf, YAP_FILENAME_MAX - 1);
do_SYSTEM_ERROR_INTERNAL(PERMISSION_ERROR_OPEN_SOURCE_SINK,
"could not open saved state");
}
return FAIL_RESTORE;
}
2017-11-11 02:15:59 +00:00
FILE *Yap_OpenRestore(const char *inpf, const char *YapLibDir) {
2015-06-19 01:30:13 +01:00
FILE *stream = NULL;
2011-08-31 21:59:30 +01:00
2016-02-13 03:11:25 +00:00
if (!inpf)
inpf = "startup.yss";
if (!YapLibDir) {
YapLibDir = YAP_LIBDIR;
}
2011-08-31 21:59:30 +01:00
OpenRestore(inpf, YapLibDir, NULL, NULL, NULL, NULL, &stream);
return stream;
}
2017-11-11 02:15:59 +00:00
static void CloseRestore(void) {
CACHE_REGS
#ifdef DEBUG_RESTORE3
2017-11-11 02:15:59 +00:00
ShowAtoms();
#endif
close_file();
LOCAL_PrologMode = UserMode;
}
#if !defined(_WIN32)
2017-11-11 02:15:59 +00:00
static int check_opcodes(OPCODE old_ops[]) {
#if USE_THREADED_CODE
2014-09-10 05:50:43 +01:00
bool have_shifted = FALSE;
op_numbers op = _Ystop;
for (op = _Ystop; op < _std_top; op++) {
if (Yap_opcode(op) != old_ops[op]) {
have_shifted = TRUE;
break;
}
}
return have_shifted;
#else
/* be conservative */
return TRUE;
#endif
}
#endif
2017-11-11 02:15:59 +00:00
static void RestoreHeap(OPCODE old_ops[] USES_REGS) {
bool heap_moved = (LOCAL_OldHeapBase != Yap_HeapBase || LOCAL_XDiff),
opcodes_moved;
Term mod = CurrentModule;
CurrentModule = PROLOG_MODULE;
#if defined(_WIN32)
/* It seems that under WIN32 opcodes may not have moved but the
remaining code may have bmoved */
opcodes_moved = TRUE;
#else
opcodes_moved = check_opcodes(old_ops);
#endif
/* opcodes_moved has side-effects and should be tried first */
if (heap_moved) {
opcodes_moved = TRUE;
2017-11-11 02:15:59 +00:00
RestoreFreeSpace(PASS_REGS1);
}
if (heap_moved || opcodes_moved) {
restore_heap();
}
/* This must be done after restore_heap */
Yap_InitAbsmi();
if (opcodes_moved) {
Yap_InitCPreds();
Yap_InitBackCPreds();
}
2017-11-11 02:15:59 +00:00
if (!(Yap_ReInitConstExps() && Yap_ReInitUnaryExps() &&
Yap_ReInitBinaryExps())) {
Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
"arithmetic operator not in saved state");
}
#ifdef DEBUG_RESTORE1
fprintf(errout, "phase 1 done\n");
#endif
CurrentModule = mod;
}
/*
* This function is called to know about the parameters of the last saved
* state
*/
2017-11-11 02:15:59 +00:00
int Yap_SavedInfo(const char *FileName, const char *YapLibDir, CELL *ATrail,
CELL *AStack, CELL *AHeap) {
2011-08-31 21:59:30 +01:00
return DO_ONLY_CODE;
CELL MyTrail, MyStack, MyHeap, MyState;
2017-11-11 02:15:59 +00:00
int mode;
2017-11-11 02:15:59 +00:00
mode = OpenRestore(FileName, YapLibDir, &MyState, &MyTrail, &MyStack, &MyHeap,
NULL);
if (mode == FAIL_RESTORE) {
return -1;
}
close_file();
2017-11-11 02:15:59 +00:00
if (!*AHeap)
*AHeap = MyHeap / 1024;
if (mode != DO_ONLY_CODE && *AStack)
*AStack = MyStack / 1024;
if (mode != DO_ONLY_CODE && *ATrail)
*ATrail = MyTrail / 1024;
return (MyState);
}
2017-11-11 02:15:59 +00:00
static void UnmarkTrEntries(USES_REGS1) {
CELL entry, *Entries;
2015-11-05 16:59:56 +00:00
/* initialize a choice point */
B = (choiceptr)LCL0;
B--;
B->cp_ap = NOCODE;
Entries = (CELL *)LOCAL_TrailBase;
while ((entry = *Entries++) != (CELL)NULL) {
if (!IsVarTerm(entry)) {
2017-11-11 02:15:59 +00:00
if (IsPairTerm(entry)) {
CELL *ent = CellPtoHeapAdjust(RepPair(entry));
register CELL flags;
flags = *ent;
ResetFlag(InUseMask, flags);
*ent = flags;
if (FlagOn((DirtyMask | ErasedMask), flags)) {
if (FlagOn(DBClMask, flags)) {
Yap_ErDBE(DBStructFlagsToDBStruct(ent));
} else {
if (flags & LogUpdMask) {
if (flags & IndexMask) {
if (FlagOn(ErasedMask, flags))
Yap_ErLogUpdIndex(ClauseFlagsToLogUpdIndex(ent));
else
Yap_CleanUpIndex(ClauseFlagsToLogUpdIndex(ent));
} else {
Yap_ErLogUpdCl(ClauseFlagsToLogUpdClause(ent));
}
} else {
Yap_ErCl(ClauseFlagsToDynamicClause(ent));
}
}
}
#ifdef MULTI_ASSIGNMENT_VARIABLES
} else /* if (IsApplTerm(d1)) */ {
2017-11-11 02:15:59 +00:00
Entries += 2;
#endif
}
}
}
B = NULL;
}
int in_limbo = FALSE;
/* cleanup any records we had in the saved state. They are now inaccessible */
2017-11-11 02:15:59 +00:00
static void FreeRecords(void) {
struct record_list *ptr;
ptr = Yap_Records;
Yap_Records = NULL;
while (ptr) {
struct record_list *optr = ptr;
Yap_ReleaseTermFromDB(ptr->dbrecord);
ptr = ptr->next_rec;
Yap_FreeCodeSpace((void *)optr);
}
}
/*
* This function is called when wanting only to restore the heap and
* associated registers
*/
2017-11-11 02:15:59 +00:00
static int Restore(char *s, char *lib_dir USES_REGS) {
int restore_mode;
2017-11-11 02:15:59 +00:00
OPCODE old_ops[_std_top + 1];
CELL MyTrail, MyStack, MyHeap, MyState;
2017-11-11 02:15:59 +00:00
if ((restore_mode = OpenRestore(s, lib_dir, &MyState, &MyTrail, &MyStack,
&MyHeap, NULL)) == FAIL_RESTORE)
return (FALSE);
Yap_ShutdownLoadForeign();
in_limbo = TRUE;
if (get_coded(restore_mode, old_ops PASS_REGS) < 0)
return FAIL_RESTORE;
restore_regs(restore_mode PASS_REGS);
in_limbo = FALSE;
/*#endif*/
RestoreHeap(old_ops PASS_REGS);
2017-11-11 02:15:59 +00:00
switch (restore_mode) {
case DO_EVERYTHING:
2017-11-11 02:15:59 +00:00
if (LOCAL_OldHeapBase != Yap_HeapBase || LOCAL_OldLCL0 != LCL0 ||
LOCAL_OldGlobalBase != (CELL *)LOCAL_GlobalBase ||
LOCAL_OldTrailBase != LOCAL_TrailBase) {
Yap_AdjustStacksAndTrail();
if (which_save == 2) {
2017-11-11 02:15:59 +00:00
Yap_AdjustRegs(2);
} else {
2017-11-11 02:15:59 +00:00
Yap_AdjustRegs(1);
}
break;
#ifdef DEBUG_RESTORE2
fprintf(errout, "phase 2 done\n");
#endif
}
break;
case DO_ONLY_CODE:
2017-11-11 02:15:59 +00:00
UnmarkTrEntries(PASS_REGS1);
Yap_InitYaamRegs(0);
break;
}
Yap_ReOpenLoadForeign();
FreeRecords();
/* restart IO */
2015-06-19 01:30:13 +01:00
// initIO();
/* reset time */
2015-11-05 16:59:56 +00:00
Yap_ReInitWTime();
#if USE_DL_MALLOC || USE_SYSTEM_MALLOC
if (!AuxSp) {
2017-11-11 02:15:59 +00:00
Yap_InitPreAllocCodeSpace(0);
}
#endif
CloseRestore();
if (which_save == 2) {
Yap_unify(ARG2, MkIntTerm(0));
}
return restore_mode;
}
2017-11-11 02:15:59 +00:00
int Yap_SavedStateRestore(char *s, char *lib_dir) {
CACHE_REGS
2017-11-11 02:15:59 +00:00
return Restore(s, lib_dir PASS_REGS);
}
2017-11-11 02:15:59 +00:00
static Int p_restore(USES_REGS1) {
int mode;
2017-11-11 02:15:59 +00:00
char s[YAP_FILENAME_MAX + 1];
Term t1 = Deref(ARG1);
2011-04-29 14:59:17 +01:00
#ifdef YAPOR
if (GLOBAL_number_workers != 1) {
2017-11-11 02:15:59 +00:00
Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
"cannot perform save: more than a worker/thread running");
return (FALSE);
}
2011-04-29 14:59:17 +01:00
#endif /* YAPOR */
#ifdef THREADS
if (GLOBAL_NOfThreads != 1) {
2017-11-11 02:15:59 +00:00
Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
"cannot perform save: more than a worker/thread running");
return (FALSE);
}
2011-04-29 14:59:17 +01:00
#endif /* THREADS */
if (!Yap_GetName(s, YAP_FILENAME_MAX, t1)) {
2017-11-11 02:15:59 +00:00
Yap_Error(TYPE_ERROR_LIST, t1, "restore/1");
return (FALSE);
}
if ((mode = Restore(s, NULL PASS_REGS)) == DO_ONLY_CODE) {
2017-11-11 02:15:59 +00:00
Yap_RestartYap(3);
}
2017-11-11 02:15:59 +00:00
return (mode != FAIL_RESTORE);
}
2017-11-11 02:15:59 +00:00
void Yap_InitSavePreds(void) {
2012-10-19 18:10:48 +01:00
Yap_InitCPred("$save", 2, p_save2, SyncPredFlag);
Yap_InitCPred("$save_program", 1, p_save_program, SyncPredFlag);
Yap_InitCPred("$restore", 1, p_restore, SyncPredFlag);
}