try to make initialisation process more robust
try to make name more robust (in case Lookup new atom fails) git-svn-id: https://yap.svn.sf.net/svnroot/yap/trunk@1254 b08c6af1-5177-4d33-ba66-4b1c6b8b522a
This commit is contained in:
parent
14f9382666
commit
f11ab09a60
@ -10,8 +10,12 @@
|
|||||||
* File: c_interface.c *
|
* File: c_interface.c *
|
||||||
* comments: c_interface primitives definition *
|
* comments: c_interface primitives definition *
|
||||||
* *
|
* *
|
||||||
* Last rev: $Date: 2005-03-01 22:25:08 $,$Author: vsc $ *
|
* Last rev: $Date: 2005-03-02 18:35:44 $,$Author: vsc $ *
|
||||||
* $Log: not supported by cvs2svn $
|
* $Log: not supported by cvs2svn $
|
||||||
|
* Revision 1.61 2005/03/01 22:25:08 vsc
|
||||||
|
* fix pruning bug
|
||||||
|
* make DL_MALLOC less enthusiastic about walking through buckets.
|
||||||
|
*
|
||||||
* Revision 1.60 2005/02/08 18:04:47 vsc
|
* Revision 1.60 2005/02/08 18:04:47 vsc
|
||||||
* library_directory may not be deterministic (usually it isn't).
|
* library_directory may not be deterministic (usually it isn't).
|
||||||
*
|
*
|
||||||
@ -95,6 +99,9 @@
|
|||||||
#include "Yap.h"
|
#include "Yap.h"
|
||||||
#include "clause.h"
|
#include "clause.h"
|
||||||
#include "yapio.h"
|
#include "yapio.h"
|
||||||
|
#if HAVE_STDARG_H
|
||||||
|
#include <stdarg.h>
|
||||||
|
#endif
|
||||||
#if _MSC_VER || defined(__MINGW32__)
|
#if _MSC_VER || defined(__MINGW32__)
|
||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
#endif
|
#endif
|
||||||
@ -165,7 +172,7 @@ X_API int STD_PROTO(YAP_StringToBuffer, (Term, char *, unsigned int));
|
|||||||
X_API Term STD_PROTO(YAP_ReadBuffer, (char *,Term *));
|
X_API Term STD_PROTO(YAP_ReadBuffer, (char *,Term *));
|
||||||
X_API Term STD_PROTO(YAP_BufferToString, (char *));
|
X_API Term STD_PROTO(YAP_BufferToString, (char *));
|
||||||
X_API Term STD_PROTO(YAP_BufferToAtomList, (char *));
|
X_API Term STD_PROTO(YAP_BufferToAtomList, (char *));
|
||||||
X_API void STD_PROTO(YAP_Error,(char *));
|
X_API void STD_PROTO(YAP_Error,(int, Term, char *, ...));
|
||||||
X_API Term STD_PROTO(YAP_RunGoal,(Term));
|
X_API Term STD_PROTO(YAP_RunGoal,(Term));
|
||||||
X_API int STD_PROTO(YAP_RestartGoal,(void));
|
X_API int STD_PROTO(YAP_RestartGoal,(void));
|
||||||
X_API int STD_PROTO(YAP_GoalHasException,(Term *));
|
X_API int STD_PROTO(YAP_GoalHasException,(Term *));
|
||||||
@ -835,9 +842,28 @@ YAP_BufferToAtomList(char *s)
|
|||||||
|
|
||||||
|
|
||||||
X_API void
|
X_API void
|
||||||
YAP_Error(char *buf)
|
YAP_Error(int myerrno, Term t, char *buf,...)
|
||||||
{
|
{
|
||||||
Yap_Error(SYSTEM_ERROR,TermNil,buf);
|
#define YAP_BUF_SIZE 512
|
||||||
|
va_list ap;
|
||||||
|
char tmpbuf[YAP_BUF_SIZE];
|
||||||
|
|
||||||
|
if (!myerrno)
|
||||||
|
myerrno = SYSTEM_ERROR;
|
||||||
|
if (t == 0L)
|
||||||
|
t = TermNil;
|
||||||
|
if (buf != NULL) {
|
||||||
|
va_start (ap, buf);
|
||||||
|
#if HAVE_VSNPRINTF
|
||||||
|
(void) vsnprintf(tmpbuf, YAP_BUF_SIZE, buf, ap);
|
||||||
|
#else
|
||||||
|
(void) vsprintf(tmpbuf, buf, ap);
|
||||||
|
#endif
|
||||||
|
va_end (ap);
|
||||||
|
} else {
|
||||||
|
tmpbuf[0] = '\0';
|
||||||
|
}
|
||||||
|
Yap_Error(myerrno,t,tmpbuf);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void myputc (int ch)
|
static void myputc (int ch)
|
||||||
@ -1046,7 +1072,9 @@ YAP_Init(YAP_init_args *yap_init)
|
|||||||
if (yap_init->SavedState != NULL ||
|
if (yap_init->SavedState != NULL ||
|
||||||
yap_init->YapPrologBootFile == NULL) {
|
yap_init->YapPrologBootFile == NULL) {
|
||||||
if (Yap_SavedInfo (yap_init->SavedState, yap_init->YapLibDir, &Trail, &Stack, &Heap) != 1) {
|
if (Yap_SavedInfo (yap_init->SavedState, yap_init->YapLibDir, &Trail, &Stack, &Heap) != 1) {
|
||||||
return(YAP_BOOT_FROM_SAVED_ERROR);
|
yap_init->ErrorNo = Yap_Error_TYPE;
|
||||||
|
yap_init->ErrorCause = Yap_ErrorMessage;
|
||||||
|
return YAP_BOOT_FROM_SAVED_ERROR;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (yap_init->TrailSize == 0) {
|
if (yap_init->TrailSize == 0) {
|
||||||
@ -1095,6 +1123,11 @@ YAP_Init(YAP_init_args *yap_init)
|
|||||||
if (yap_init->SavedState != NULL ||
|
if (yap_init->SavedState != NULL ||
|
||||||
yap_init->YapPrologBootFile == NULL) {
|
yap_init->YapPrologBootFile == NULL) {
|
||||||
restore_result = Yap_Restore(yap_init->SavedState, yap_init->YapLibDir);
|
restore_result = Yap_Restore(yap_init->SavedState, yap_init->YapLibDir);
|
||||||
|
if (restore_result == FAIL_RESTORE) {
|
||||||
|
yap_init->ErrorNo = Yap_Error_TYPE;
|
||||||
|
yap_init->ErrorCause = Yap_ErrorMessage;
|
||||||
|
return YAP_BOOT_FROM_SAVED_ERROR;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
restore_result = FAIL_RESTORE;
|
restore_result = FAIL_RESTORE;
|
||||||
}
|
}
|
||||||
@ -1134,21 +1167,25 @@ YAP_Init(YAP_init_args *yap_init)
|
|||||||
}
|
}
|
||||||
if (yap_init->SavedState != NULL ||
|
if (yap_init->SavedState != NULL ||
|
||||||
yap_init->YapPrologBootFile == NULL) {
|
yap_init->YapPrologBootFile == NULL) {
|
||||||
if (restore_result == FAIL_RESTORE)
|
if (restore_result == FAIL_RESTORE) {
|
||||||
return(YAP_BOOT_FROM_SAVED_ERROR);
|
yap_init->ErrorNo = Yap_Error_TYPE;
|
||||||
|
yap_init->ErrorCause = Yap_ErrorMessage;
|
||||||
|
return YAP_BOOT_FROM_SAVED_ERROR;
|
||||||
|
}
|
||||||
if (restore_result == DO_ONLY_CODE) {
|
if (restore_result == DO_ONLY_CODE) {
|
||||||
return(YAP_BOOT_FROM_SAVED_CODE);
|
return YAP_BOOT_FROM_SAVED_CODE;
|
||||||
} else {
|
} else {
|
||||||
return(YAP_BOOT_FROM_SAVED_STACKS);
|
return YAP_BOOT_FROM_SAVED_STACKS;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return(YAP_BOOT_FROM_PROLOG);
|
return YAP_BOOT_FROM_PROLOG;
|
||||||
}
|
}
|
||||||
|
|
||||||
X_API Int
|
X_API Int
|
||||||
YAP_FastInit(char saved_state[])
|
YAP_FastInit(char saved_state[])
|
||||||
{
|
{
|
||||||
YAP_init_args init_args;
|
YAP_init_args init_args;
|
||||||
|
Int out;
|
||||||
|
|
||||||
init_args.SavedState = saved_state;
|
init_args.SavedState = saved_state;
|
||||||
init_args.HeapSize = 0;
|
init_args.HeapSize = 0;
|
||||||
@ -1166,8 +1203,13 @@ YAP_FastInit(char saved_state[])
|
|||||||
init_args.PrologShouldHandleInterrupts = FALSE;
|
init_args.PrologShouldHandleInterrupts = FALSE;
|
||||||
init_args.Argc = 0;
|
init_args.Argc = 0;
|
||||||
init_args.Argv = NULL;
|
init_args.Argv = NULL;
|
||||||
|
init_args.ErrorNo = 0;
|
||||||
return(YAP_Init(&init_args));
|
init_args.ErrorCause = NULL;
|
||||||
|
out = YAP_Init(&init_args);
|
||||||
|
if (out == YAP_BOOT_FROM_SAVED_ERROR) {
|
||||||
|
Yap_Error(init_args.ErrorNo,TermNil,init_args.ErrorCause);
|
||||||
|
}
|
||||||
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
X_API void
|
X_API void
|
||||||
|
19
C/errors.c
19
C/errors.c
@ -372,7 +372,10 @@ Yap_Error(yap_error_number type, Term where, char *format,...)
|
|||||||
int psize = YAP_BUF_SIZE;
|
int psize = YAP_BUF_SIZE;
|
||||||
|
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
/* fprintf(stderr,"***** Processing Error %d (%x,%x) %s***\n", type, ActiveSignals,Yap_PrologMode,format);*/
|
if (Yap_heap_regs && !(Yap_PrologMode & BootMode))
|
||||||
|
fprintf(stderr,"***** Processing Error %d (%x,%x) %s***\n", type, ActiveSignals,Yap_PrologMode,format);
|
||||||
|
else
|
||||||
|
fprintf(stderr,"***** Processing Error %d (%x) %s***\n", type,Yap_PrologMode,format);
|
||||||
#endif
|
#endif
|
||||||
if (type == INTERRUPT_ERROR) {
|
if (type == INTERRUPT_ERROR) {
|
||||||
fprintf(stderr,"%% YAP exiting: cannot handle signal %d\n",
|
fprintf(stderr,"%% YAP exiting: cannot handle signal %d\n",
|
||||||
@ -419,9 +422,9 @@ Yap_Error(yap_error_number type, Term where, char *format,...)
|
|||||||
tmpbuf[0] = '\0';
|
tmpbuf[0] = '\0';
|
||||||
}
|
}
|
||||||
if (Yap_PrologMode == UserCCallMode) {
|
if (Yap_PrologMode == UserCCallMode) {
|
||||||
fprintf(stderr,"%% OOOPS in USER C-CODE: %s.\n",tmpbuf);
|
fprintf(stderr,"%% YAP OOOPS in USER C-CODE: %s.\n",tmpbuf);
|
||||||
} else {
|
} else {
|
||||||
fprintf(stderr,"%% OOOPS: %s.\n",tmpbuf);
|
fprintf(stderr,"%% YAP OOOPS: %s.\n",tmpbuf);
|
||||||
}
|
}
|
||||||
error_exit_yap (1);
|
error_exit_yap (1);
|
||||||
}
|
}
|
||||||
@ -461,7 +464,7 @@ Yap_Error(yap_error_number type, Term where, char *format,...)
|
|||||||
va_end (ap);
|
va_end (ap);
|
||||||
if (Yap_PrologMode & BootMode) {
|
if (Yap_PrologMode & BootMode) {
|
||||||
/* crash in flames! */
|
/* crash in flames! */
|
||||||
fprintf(stderr,"%% Fatal Error: %s exiting....\n",tmpbuf);
|
fprintf(stderr,"%% YAP Fatal Error: %s exiting....\n",tmpbuf);
|
||||||
error_exit_yap (1);
|
error_exit_yap (1);
|
||||||
}
|
}
|
||||||
#ifdef DEBUGX
|
#ifdef DEBUGX
|
||||||
@ -472,8 +475,12 @@ Yap_Error(yap_error_number type, Term where, char *format,...)
|
|||||||
{
|
{
|
||||||
fprintf(stderr,"%% Internal YAP Error: %s exiting....\n",tmpbuf);
|
fprintf(stderr,"%% Internal YAP Error: %s exiting....\n",tmpbuf);
|
||||||
serious = TRUE;
|
serious = TRUE;
|
||||||
detect_bug_location(P, FIND_PRED_FROM_ANYWHERE, tmpbuf, YAP_BUF_SIZE);
|
if (Yap_PrologMode & BootMode) {
|
||||||
fprintf(stderr,"%% Bug found while executing %s\n",tmpbuf);
|
fprintf(stderr,"%% YAP crashed while booting %s\n",tmpbuf);
|
||||||
|
} else {
|
||||||
|
detect_bug_location(P, FIND_PRED_FROM_ANYWHERE, tmpbuf, YAP_BUF_SIZE);
|
||||||
|
fprintf(stderr,"%% Bug found while executing %s\n",tmpbuf);
|
||||||
|
}
|
||||||
error_exit_yap (1);
|
error_exit_yap (1);
|
||||||
}
|
}
|
||||||
case FATAL_ERROR:
|
case FATAL_ERROR:
|
||||||
|
5
C/exec.c
5
C/exec.c
@ -1450,11 +1450,14 @@ JumpToEnv(Term t) {
|
|||||||
#ifdef TABLING
|
#ifdef TABLING
|
||||||
abolish_incomplete_subgoals(B);
|
abolish_incomplete_subgoals(B);
|
||||||
#endif /* TABLING */
|
#endif /* TABLING */
|
||||||
return(FALSE);
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
Int
|
Int
|
||||||
Yap_JumpToEnv(Term t) {
|
Yap_JumpToEnv(Term t) {
|
||||||
|
if (Yap_PrologMode & BootMode) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
return JumpToEnv(t);
|
return JumpToEnv(t);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
338
C/save.c
338
C/save.c
@ -81,7 +81,7 @@ static char end_msg[256] ="*** End of YAP saved state *****";
|
|||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
STATIC_PROTO(void myread, (int, char *, Int));
|
STATIC_PROTO(int myread, (int, char *, Int));
|
||||||
STATIC_PROTO(void mywrite, (int, char *, Int));
|
STATIC_PROTO(void mywrite, (int, char *, Int));
|
||||||
STATIC_PROTO(int open_file, (char *, int));
|
STATIC_PROTO(int open_file, (char *, int));
|
||||||
STATIC_PROTO(void close_file, (void));
|
STATIC_PROTO(void close_file, (void));
|
||||||
@ -99,13 +99,13 @@ STATIC_PROTO(Int do_save, (int));
|
|||||||
STATIC_PROTO(Int p_save, (void));
|
STATIC_PROTO(Int p_save, (void));
|
||||||
STATIC_PROTO(Int p_save_program, (void));
|
STATIC_PROTO(Int p_save_program, (void));
|
||||||
STATIC_PROTO(int check_header, (CELL *, CELL *, CELL *, CELL *));
|
STATIC_PROTO(int check_header, (CELL *, CELL *, CELL *, CELL *));
|
||||||
STATIC_PROTO(void get_heap_info, (void));
|
STATIC_PROTO(int get_heap_info, (void));
|
||||||
STATIC_PROTO(void get_regs, (int));
|
STATIC_PROTO(int get_regs, (int));
|
||||||
STATIC_PROTO(void get_insts, (OPCODE []));
|
STATIC_PROTO(int get_insts, (OPCODE []));
|
||||||
STATIC_PROTO(void get_hash, (void));
|
STATIC_PROTO(int get_hash, (void));
|
||||||
STATIC_PROTO(void CopyCode, (void));
|
STATIC_PROTO(int CopyCode, (void));
|
||||||
STATIC_PROTO(void CopyStacks, (void));
|
STATIC_PROTO(int CopyStacks, (void));
|
||||||
STATIC_PROTO(void get_coded, (int, OPCODE []));
|
STATIC_PROTO(int get_coded, (int, OPCODE []));
|
||||||
STATIC_PROTO(void restore_codes, (void));
|
STATIC_PROTO(void restore_codes, (void));
|
||||||
STATIC_PROTO(Term AdjustDBTerm, (Term, Term *));
|
STATIC_PROTO(Term AdjustDBTerm, (Term, Term *));
|
||||||
STATIC_PROTO(void RestoreDB, (DBEntry *));
|
STATIC_PROTO(void RestoreDB, (DBEntry *));
|
||||||
@ -152,23 +152,45 @@ LightBug(s)
|
|||||||
|
|
||||||
#endif /* LIGHT */
|
#endif /* LIGHT */
|
||||||
|
|
||||||
|
static int
|
||||||
|
do_system_error(yap_error_number etype, const char *msg)
|
||||||
|
{
|
||||||
|
#if HAVE_SNPRINTF
|
||||||
|
#if HAVE_STRERROR
|
||||||
|
snprintf(Yap_ErrorSay,MAX_ERROR_MSG_SIZE,"%s (%s when reading %s)", msg, strerror(errno), Yap_FileNameBuf);
|
||||||
|
#else
|
||||||
|
snprintf(Yap_ErrorSay,MAX_ERROR_MSG_SIZE,"%s, (system error %d when reading %s)",msg,errno,Yap_FileNameBuf);
|
||||||
|
#endif
|
||||||
|
#else
|
||||||
|
#if HAVE_STRERROR
|
||||||
|
sprintf(Yap_ErrorSay,"%s, (%s when reading %s)",msg,strerror(errno),Yap_FileNameBuf);
|
||||||
|
#else
|
||||||
|
sprintf(Yap_ErrorSay,"%s, (system error %d when reading %s)",msg,errno,Yap_FileNameBuf);
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
Yap_ErrorMessage = Yap_ErrorSay;
|
||||||
|
Yap_Error_TYPE = etype;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
#if SHORT_INTS
|
#if SHORT_INTS
|
||||||
|
|
||||||
#ifdef M_WILLIAMS
|
#ifdef M_WILLIAMS
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static void
|
static int
|
||||||
myread(int fd, char *buff, Int len)
|
myread(int fd, char *buff, Int len)
|
||||||
{
|
{
|
||||||
while (len > 16000) {
|
while (len > 0) {
|
||||||
int nchars = read(fd, buff, 16000);
|
int nchars = read(fd, buff, len);
|
||||||
if (nchars <= 0)
|
if (nchars < 0) {
|
||||||
Yap_Error(FATAL_ERROR,TermNil,"bad saved state, system corrupted");
|
return do_system_error(PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM, "bad read on saved state", );
|
||||||
len -= 16000;
|
}
|
||||||
buff += 16000;
|
len -= nchars;
|
||||||
|
buff += nchars;
|
||||||
}
|
}
|
||||||
read(fd, buff, (unsigned) len);
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -185,17 +207,18 @@ mywrite(int fd, char *buff, Int len)
|
|||||||
#else /* SHORT_INTS */
|
#else /* SHORT_INTS */
|
||||||
|
|
||||||
inline static
|
inline static
|
||||||
void myread(int fd, char *buffer, Int len) {
|
int myread(int fd, char *buffer, Int len) {
|
||||||
int nread;
|
int nread;
|
||||||
|
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
nread = read(fd, buffer, (int)len);
|
nread = read(fd, buffer, (int)len);
|
||||||
if (nread < 1) {
|
if (nread < 1) {
|
||||||
Yap_Error(FATAL_ERROR,TermNil,"bad saved state, system corrupted");
|
return do_system_error(PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM,"bad read on saved state");
|
||||||
}
|
}
|
||||||
buffer += nread;
|
buffer += nread;
|
||||||
len -= nread;
|
len -= nread;
|
||||||
}
|
}
|
||||||
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline static
|
inline static
|
||||||
@ -204,8 +227,9 @@ void mywrite(int fd, char *buff, Int len) {
|
|||||||
|
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
nwritten = (Int)write(fd, buff, (int)len);
|
nwritten = (Int)write(fd, buff, (int)len);
|
||||||
if (nwritten == -1) {
|
if (nwritten < 0) {
|
||||||
Yap_Error(SYSTEM_ERROR,TermNil,"write error while saving");
|
Yap_ErrorMessage = "bad write on saved state";
|
||||||
|
Yap_Error(SYSTEM_ERROR,TermNil,Yap_ErrorMessage);
|
||||||
}
|
}
|
||||||
buff += nwritten;
|
buff += nwritten;
|
||||||
len -= nwritten;
|
len -= nwritten;
|
||||||
@ -263,12 +287,12 @@ open_file(char *my_file, int flag)
|
|||||||
#endif /* M_WILLIAMS */
|
#endif /* M_WILLIAMS */
|
||||||
{
|
{
|
||||||
splfild = 0; /* We do not have an open file */
|
splfild = 0; /* We do not have an open file */
|
||||||
return(-1);
|
return -1;
|
||||||
}
|
}
|
||||||
#ifdef undf0
|
#ifdef undf0
|
||||||
fprintf(errout, "Opened file %s\n", my_file);
|
fprintf(errout, "Opened file %s\n", my_file);
|
||||||
#endif
|
#endif
|
||||||
return(splfild);
|
return splfild;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -311,12 +335,12 @@ get_header_cell(void)
|
|||||||
int count = 0, n;
|
int count = 0, n;
|
||||||
while (count < sizeof(CELL)) {
|
while (count < sizeof(CELL)) {
|
||||||
if ((n = read(splfild, &l, sizeof(CELL)-count)) < 0) {
|
if ((n = read(splfild, &l, sizeof(CELL)-count)) < 0) {
|
||||||
Yap_ErrorMessage = "corrupt saved state (too short)";
|
do_system_error(PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM,"failed to read saved state header");
|
||||||
return(0L);
|
return 0L;
|
||||||
}
|
}
|
||||||
count += n;
|
count += n;
|
||||||
}
|
}
|
||||||
return(l);
|
return l;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* gets a pointer to cell from a file */
|
/* gets a pointer to cell from a file */
|
||||||
@ -524,7 +548,7 @@ do_save(int mode) {
|
|||||||
Term t1 = Deref(ARG1);
|
Term t1 = Deref(ARG1);
|
||||||
if (!Yap_GetName(Yap_FileNameBuf, YAP_FILENAME_MAX, t1)) {
|
if (!Yap_GetName(Yap_FileNameBuf, YAP_FILENAME_MAX, t1)) {
|
||||||
Yap_Error(TYPE_ERROR_LIST,t1,"save/1");
|
Yap_Error(TYPE_ERROR_LIST,t1,"save/1");
|
||||||
return(FALSE);
|
return FALSE;
|
||||||
}
|
}
|
||||||
Yap_CloseStreams(TRUE);
|
Yap_CloseStreams(TRUE);
|
||||||
if ((splfild = open_file(Yap_FileNameBuf, O_WRONLY | O_CREAT)) < 0) {
|
if ((splfild = open_file(Yap_FileNameBuf, O_WRONLY | O_CREAT)) < 0) {
|
||||||
@ -576,7 +600,7 @@ static Int
|
|||||||
p_save_program(void)
|
p_save_program(void)
|
||||||
{
|
{
|
||||||
which_save = 0;
|
which_save = 0;
|
||||||
return(do_save(DO_ONLY_CODE));
|
return do_save(DO_ONLY_CODE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Now, to restore the saved code */
|
/* Now, to restore the saved code */
|
||||||
@ -593,8 +617,8 @@ check_header(CELL *info, CELL *ATrail, CELL *AStack, CELL *AHeap)
|
|||||||
/* skip the first line */
|
/* skip the first line */
|
||||||
do {
|
do {
|
||||||
if (read(splfild, pp, 1) < 0) {
|
if (read(splfild, pp, 1) < 0) {
|
||||||
Yap_ErrorMessage = "corrupt saved state (failed to read first line)";
|
do_system_error(PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM,"failed to scan first line from saved state");
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
}
|
}
|
||||||
} while (pp[0] != 1);
|
} while (pp[0] != 1);
|
||||||
/* now check the version */
|
/* now check the version */
|
||||||
@ -603,97 +627,126 @@ check_header(CELL *info, CELL *ATrail, CELL *AStack, CELL *AHeap)
|
|||||||
int count = 0, n, to_read = Unsigned(strlen(msg) + 1);
|
int count = 0, n, to_read = Unsigned(strlen(msg) + 1);
|
||||||
while (count < to_read) {
|
while (count < to_read) {
|
||||||
if ((n = read(splfild, pp, to_read-count)) < 0) {
|
if ((n = read(splfild, pp, to_read-count)) < 0) {
|
||||||
Yap_ErrorMessage = "corrupt saved state (header too short)";
|
do_system_error(PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM,"failed to scan version info from saved state");
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
}
|
}
|
||||||
count += n;
|
count += n;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (pp[0] != 'Y' && pp[1] != 'A' && pp[0] != 'P') {
|
if (pp[0] != 'Y' && pp[1] != 'A' && pp[0] != 'P') {
|
||||||
Yap_ErrorMessage = "corrupt saved state (should say YAP)";
|
Yap_ErrorMessage = Yap_ErrorSay;
|
||||||
return(FAIL_RESTORE);
|
strncpy(Yap_ErrorMessage, "saved state ", MAX_ERROR_MSG_SIZE);
|
||||||
|
strncat(Yap_ErrorMessage, Yap_FileNameBuf, MAX_ERROR_MSG_SIZE);
|
||||||
|
strncat(Yap_ErrorMessage, " failed to match `YAP\'", MAX_ERROR_MSG_SIZE);
|
||||||
|
Yap_Error_TYPE = CONSISTENCY_ERROR;
|
||||||
|
return FAIL_RESTORE;
|
||||||
}
|
}
|
||||||
if (strcmp(pp, msg) != 0) {
|
if (strcmp(pp, msg) != 0) {
|
||||||
Yap_ErrorMessage = "corrupt saved state (different version of YAP)";
|
Yap_ErrorMessage = Yap_ErrorSay;
|
||||||
return(FAIL_RESTORE);
|
strncpy(Yap_ErrorMessage, "saved state ", MAX_ERROR_MSG_SIZE);
|
||||||
|
strncat(Yap_ErrorMessage, Yap_FileNameBuf, MAX_ERROR_MSG_SIZE);
|
||||||
|
strncat(Yap_ErrorMessage, " failed to match version ID", MAX_ERROR_MSG_SIZE);
|
||||||
|
Yap_Error_TYPE = CONSISTENCY_ERROR;
|
||||||
|
return FAIL_RESTORE;
|
||||||
}
|
}
|
||||||
/* check info on header */
|
/* check info on header */
|
||||||
/* ignore info on saved state */
|
/* ignore info on saved state */
|
||||||
*info = get_header_cell();
|
*info = get_header_cell();
|
||||||
if (Yap_ErrorMessage)
|
if (Yap_ErrorMessage)
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
/* check the restore mode */
|
/* check the restore mode */
|
||||||
mode = get_header_cell();
|
mode = get_header_cell();
|
||||||
if (Yap_ErrorMessage)
|
if (Yap_ErrorMessage)
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
if (mode != DO_EVERYTHING && mode != DO_ONLY_CODE) {
|
if (mode != DO_EVERYTHING && mode != DO_ONLY_CODE) {
|
||||||
Yap_ErrorMessage = "corrupt saved state (bad type)";
|
return FAIL_RESTORE;
|
||||||
return(FAIL_RESTORE);
|
|
||||||
}
|
}
|
||||||
/* ignore info on stacks size */
|
/* ignore info on stacks size */
|
||||||
*AHeap = get_header_cell();
|
*AHeap = get_header_cell();
|
||||||
|
if (Yap_ErrorMessage) {
|
||||||
|
return FAIL_RESTORE;
|
||||||
|
}
|
||||||
*AStack = get_header_cell();
|
*AStack = get_header_cell();
|
||||||
|
if (Yap_ErrorMessage) {
|
||||||
|
return FAIL_RESTORE;
|
||||||
|
}
|
||||||
*ATrail = get_header_cell();
|
*ATrail = get_header_cell();
|
||||||
if (Yap_ErrorMessage)
|
if (Yap_ErrorMessage) {
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
|
}
|
||||||
/* now, check whether we got enough enough space to load the
|
/* now, check whether we got enough enough space to load the
|
||||||
saved space */
|
saved space */
|
||||||
hp_size = get_cell();
|
hp_size = get_cell();
|
||||||
if (Yap_ErrorMessage)
|
if (Yap_ErrorMessage)
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
while (Yap_HeapBase != NULL &&
|
while (Yap_HeapBase != NULL &&
|
||||||
hp_size > Unsigned(HeapLim) - Unsigned(Yap_HeapBase)) {
|
hp_size > Unsigned(HeapLim) - Unsigned(Yap_HeapBase)) {
|
||||||
if(!Yap_growheap(FALSE, hp_size, NULL)) {
|
if(!Yap_growheap(FALSE, hp_size, NULL)) {
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (mode == DO_EVERYTHING) {
|
if (mode == DO_EVERYTHING) {
|
||||||
lc_size = get_cell();
|
lc_size = get_cell();
|
||||||
if (Yap_ErrorMessage)
|
if (Yap_ErrorMessage)
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
gb_size=get_cell();
|
gb_size=get_cell();
|
||||||
if (Yap_ErrorMessage)
|
if (Yap_ErrorMessage)
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
if (Yap_HeapBase != NULL && lc_size+gb_size > Unsigned(Yap_LocalBase) - Unsigned(Yap_GlobalBase)) {
|
if (Yap_HeapBase != NULL && lc_size+gb_size > Unsigned(Yap_LocalBase) - Unsigned(Yap_GlobalBase)) {
|
||||||
Yap_ErrorMessage = "not enough stack space for restore";
|
if (Yap_ErrorMessage != NULL)
|
||||||
return(FAIL_RESTORE);
|
Yap_ErrorMessage = "could not allocate enough stack space";
|
||||||
|
return FAIL_RESTORE;
|
||||||
}
|
}
|
||||||
if (Yap_HeapBase != NULL && (tr_size = get_cell()) > Unsigned(Yap_TrailTop) - Unsigned(Yap_TrailBase)) {
|
if (Yap_HeapBase != NULL && (tr_size = get_cell()) > Unsigned(Yap_TrailTop) - Unsigned(Yap_TrailBase)) {
|
||||||
Yap_ErrorMessage = "not enough trail space for restore";
|
if (Yap_ErrorMessage != NULL)
|
||||||
return(FAIL_RESTORE);
|
Yap_ErrorMessage = "could not allocate enough trail space";
|
||||||
|
return FAIL_RESTORE;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* skip cell size */
|
/* skip cell size */
|
||||||
get_header_cell();
|
get_header_cell();
|
||||||
if (Yap_ErrorMessage)
|
if (Yap_ErrorMessage)
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
get_header_cell();
|
get_header_cell();
|
||||||
if (Yap_ErrorMessage)
|
if (Yap_ErrorMessage)
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
get_header_cell();
|
get_header_cell();
|
||||||
if (Yap_ErrorMessage)
|
if (Yap_ErrorMessage)
|
||||||
return(FAIL_RESTORE);
|
return FAIL_RESTORE;
|
||||||
}
|
}
|
||||||
return(mode);
|
return(mode);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Gets the state of the heap, and evaluates the related variables */
|
/* Gets the state of the heap, and evaluates the related variables */
|
||||||
static void
|
static int
|
||||||
get_heap_info(void)
|
get_heap_info(void)
|
||||||
{
|
{
|
||||||
OldHeapBase = (ADDR) get_cellptr();
|
OldHeapBase = (ADDR) get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
OldHeapTop = (ADDR) get_cellptr();
|
OldHeapTop = (ADDR) get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
OldHeapUsed = (Int) get_cell();
|
OldHeapUsed = (Int) get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
FreeBlocks = (BlockHeader *) get_cellptr();
|
FreeBlocks = (BlockHeader *) get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
AuxSp = get_cellptr();
|
AuxSp = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
AuxTop = (ADDR)get_cellptr();
|
AuxTop = (ADDR)get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
HDiff = Unsigned(Yap_HeapBase) - Unsigned(OldHeapBase);
|
HDiff = Unsigned(Yap_HeapBase) - Unsigned(OldHeapBase);
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Gets the register array */
|
/* Gets the register array */
|
||||||
/* Saves the old bases for the work areas */
|
/* Saves the old bases for the work areas */
|
||||||
/* and evaluates the difference from the old areas to the new ones */
|
/* and evaluates the difference from the old areas to the new ones */
|
||||||
static void
|
static int
|
||||||
get_regs(int flag)
|
get_regs(int flag)
|
||||||
{
|
{
|
||||||
CELL *NewGlobalBase = (CELL *)Yap_GlobalBase;
|
CELL *NewGlobalBase = (CELL *)Yap_GlobalBase;
|
||||||
@ -702,46 +755,95 @@ get_regs(int flag)
|
|||||||
|
|
||||||
/* Get regs */
|
/* Get regs */
|
||||||
compile_arrays = (int)get_cell();
|
compile_arrays = (int)get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
if (flag == DO_EVERYTHING) {
|
if (flag == DO_EVERYTHING) {
|
||||||
CP = (yamop *)get_cellptr();
|
CP = (yamop *)get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
ENV = get_cellptr();
|
ENV = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
ASP = get_cellptr();
|
ASP = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
/* N = get_cell(); */
|
/* N = get_cell(); */
|
||||||
H0 = get_cellptr();
|
H0 = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
LCL0 = get_cellptr();
|
LCL0 = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
H = get_cellptr();
|
H = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
HB = get_cellptr();
|
HB = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
B = (choiceptr)get_cellptr();
|
B = (choiceptr)get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
TR = (tr_fr_ptr)get_cellptr();
|
TR = (tr_fr_ptr)get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
YENV = get_cellptr();
|
YENV = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
S = get_cellptr();
|
S = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
P = (yamop *)get_cellptr();
|
P = (yamop *)get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
CreepFlag = get_cell();
|
CreepFlag = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
#ifdef COROUTINING
|
#ifdef COROUTINING
|
||||||
DelayedVars = get_cell();
|
DelayedVars = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
CurrentModule = get_cell();
|
CurrentModule = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
if (flag == DO_EVERYTHING) {
|
if (flag == DO_EVERYTHING) {
|
||||||
#ifdef COROUTINING
|
#ifdef COROUTINING
|
||||||
WokenGoals = get_cell();
|
WokenGoals = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
#endif
|
#endif
|
||||||
#ifdef DEPTH_LIMIT
|
#ifdef DEPTH_LIMIT
|
||||||
DEPTH = get_cell();
|
DEPTH = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
/* Get the old bases */
|
/* Get the old bases */
|
||||||
OldXREGS = get_cellptr();
|
OldXREGS = get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
which_save = get_cell();
|
which_save = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
XDiff = (CELL)XREGS - (CELL)OldXREGS;
|
XDiff = (CELL)XREGS - (CELL)OldXREGS;
|
||||||
get_heap_info();
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
|
if (get_heap_info() < 0)
|
||||||
|
return -1;
|
||||||
if (flag == DO_EVERYTHING) {
|
if (flag == DO_EVERYTHING) {
|
||||||
ARG1 = get_cell();
|
ARG1 = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
if (which_save == 2) {
|
if (which_save == 2) {
|
||||||
ARG2 = get_cell();
|
ARG2 = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
/* get old trail base */
|
/* get old trail base */
|
||||||
OldTrailBase = (ADDR)get_cellptr();
|
OldTrailBase = (ADDR)get_cellptr();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
/* Save the old register where we can easily access them */
|
/* Save the old register where we can easily access them */
|
||||||
OldASP = ASP;
|
OldASP = ASP;
|
||||||
OldLCL0 = LCL0;
|
OldLCL0 = LCL0;
|
||||||
@ -754,54 +856,71 @@ get_regs(int flag)
|
|||||||
Yap_GlobalBase = (ADDR)NewGlobalBase;
|
Yap_GlobalBase = (ADDR)NewGlobalBase;
|
||||||
LCL0 = NewLCL0;
|
LCL0 = NewLCL0;
|
||||||
}
|
}
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the old opcodes and place them in a hash table */
|
/* Get the old opcodes and place them in a hash table */
|
||||||
static void
|
static int
|
||||||
get_insts(OPCODE old_ops[])
|
get_insts(OPCODE old_ops[])
|
||||||
{
|
{
|
||||||
myread(splfild, (char *)old_ops, sizeof(OPCODE)*(_std_top+1));
|
return myread(splfild, (char *)old_ops, sizeof(OPCODE)*(_std_top+1));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the old atoms hash table */
|
/* Get the old atoms hash table */
|
||||||
static void
|
static int
|
||||||
get_hash(void)
|
get_hash(void)
|
||||||
{
|
{
|
||||||
myread(splfild, Yap_chtype , NUMBER_OF_CHARS);
|
return myread(splfild, Yap_chtype , NUMBER_OF_CHARS);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Copy all of the old code to the new Heap */
|
/* Copy all of the old code to the new Heap */
|
||||||
static void
|
static int
|
||||||
CopyCode(void)
|
CopyCode(void)
|
||||||
{
|
{
|
||||||
#if defined(YAPOR) || defined(TABLING)
|
#if defined(YAPOR) || defined(TABLING)
|
||||||
/* skip the local and global data structures */
|
/* skip the local and global data structures */
|
||||||
CELL j = get_cell();
|
CELL j = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
if (j != Unsigned(&GLOBAL) - Unsigned(Yap_HeapBase)) {
|
if (j != Unsigned(&GLOBAL) - Unsigned(Yap_HeapBase)) {
|
||||||
Yap_Error(FATAL_ERROR,TermNil,"bad saved state (code space size does not match)");
|
Yap_ErrorMessage = "code space size does not match saved state";
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
myread(splfild, (char *) Yap_HeapBase, j);
|
if (myread(splfild, (char *) Yap_HeapBase, j) < 0)
|
||||||
|
return -1;
|
||||||
#ifdef USE_HEAP
|
#ifdef USE_HEAP
|
||||||
j = get_cell();
|
j = get_cell();
|
||||||
myread(splfild, (char *) &HashChain, j);
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
|
if (myread(splfild, (char *) &HashChain, j) < 0)
|
||||||
|
return -1;
|
||||||
#else
|
#else
|
||||||
j = get_cell();
|
j = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
if (j != Unsigned(BaseAllocArea) - Unsigned(&HashChain)) {
|
if (j != Unsigned(BaseAllocArea) - Unsigned(&HashChain)) {
|
||||||
Yap_Error(FATAL_ERROR,TermNil,"bad saved state (Base to Hash does not match)");
|
Yap_ErrorMessage = "Base to Hash does not match saved state";
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
myread(splfild, (char *) &HashChain, j);
|
if (myread(splfild, (char *) &HashChain, j) < 0)
|
||||||
|
return -1;
|
||||||
j = get_cell();
|
j = get_cell();
|
||||||
myread(splfild, (char *) TopAllocBlockArea, j);
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
|
if (myread(splfild, (char *) TopAllocBlockArea, j) < 0)
|
||||||
|
return -1;
|
||||||
#endif
|
#endif
|
||||||
#else
|
#else
|
||||||
myread(splfild, (char *) Yap_HeapBase,
|
if (myread(splfild, (char *) Yap_HeapBase,
|
||||||
(Unsigned(OldHeapTop) - Unsigned(OldHeapBase)));
|
(Unsigned(OldHeapTop) - Unsigned(OldHeapBase))) < 0)
|
||||||
|
return -1;
|
||||||
#endif /* YAPOR || TABLING */
|
#endif /* YAPOR || TABLING */
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Copy the local and global stack and also the trail to their new home */
|
/* Copy the local and global stack and also the trail to their new home */
|
||||||
/* In REGS we still have nonadjusted values !! */
|
/* In REGS we still have nonadjusted values !! */
|
||||||
static void
|
static int
|
||||||
CopyStacks(void)
|
CopyStacks(void)
|
||||||
{
|
{
|
||||||
Int j;
|
Int j;
|
||||||
@ -809,16 +928,20 @@ CopyStacks(void)
|
|||||||
|
|
||||||
j = Unsigned(OldLCL0) - Unsigned(ASP);
|
j = Unsigned(OldLCL0) - Unsigned(ASP);
|
||||||
NewASP = (char *) (Unsigned(ASP) + (Unsigned(LCL0) - Unsigned(OldLCL0)));
|
NewASP = (char *) (Unsigned(ASP) + (Unsigned(LCL0) - Unsigned(OldLCL0)));
|
||||||
myread(splfild, (char *) NewASP, j);
|
if (myread(splfild, (char *) NewASP, j) < 0)
|
||||||
|
return -1;
|
||||||
j = Unsigned(H) - Unsigned(OldGlobalBase);
|
j = Unsigned(H) - Unsigned(OldGlobalBase);
|
||||||
myread(splfild, (char *) Yap_GlobalBase, j);
|
if (myread(splfild, (char *) Yap_GlobalBase, j) < 0)
|
||||||
|
return -1;
|
||||||
j = Unsigned(TR) - Unsigned(OldTrailBase);
|
j = Unsigned(TR) - Unsigned(OldTrailBase);
|
||||||
myread(splfild, Yap_TrailBase, j);
|
if (myread(splfild, Yap_TrailBase, j))
|
||||||
|
return -1;
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Copy the local and global stack and also the trail to their new home */
|
/* Copy the local and global stack and also the trail to their new home */
|
||||||
/* In REGS we still have nonadjusted values !! */
|
/* In REGS we still have nonadjusted values !! */
|
||||||
static void
|
static int
|
||||||
CopyTrailEntries(void)
|
CopyTrailEntries(void)
|
||||||
{
|
{
|
||||||
CELL entry, *Entries;
|
CELL entry, *Entries;
|
||||||
@ -826,31 +949,44 @@ CopyTrailEntries(void)
|
|||||||
Entries = (CELL *)Yap_TrailBase;
|
Entries = (CELL *)Yap_TrailBase;
|
||||||
do {
|
do {
|
||||||
*Entries++ = entry = get_cell();
|
*Entries++ = entry = get_cell();
|
||||||
|
if (Yap_ErrorMessage)
|
||||||
|
return -1;
|
||||||
} while ((CODEADDR)entry != NULL);
|
} while ((CODEADDR)entry != NULL);
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* get things which are saved in the file */
|
/* get things which are saved in the file */
|
||||||
static void
|
static int
|
||||||
get_coded(int flag, OPCODE old_ops[])
|
get_coded(int flag, OPCODE old_ops[])
|
||||||
{
|
{
|
||||||
char my_end_msg[256];
|
char my_end_msg[256];
|
||||||
|
|
||||||
get_regs(flag);
|
if (get_regs(flag) < 0)
|
||||||
get_insts(old_ops);
|
return -1;
|
||||||
get_hash();
|
if (get_insts(old_ops) < 0)
|
||||||
CopyCode();
|
return -1;
|
||||||
|
if (get_hash() < 0)
|
||||||
|
return -1;
|
||||||
|
if (CopyCode() < 0)
|
||||||
|
return -1;
|
||||||
switch (flag) {
|
switch (flag) {
|
||||||
case DO_EVERYTHING:
|
case DO_EVERYTHING:
|
||||||
CopyStacks();
|
if (CopyStacks() < 0)
|
||||||
|
return -1;
|
||||||
break;
|
break;
|
||||||
case DO_ONLY_CODE:
|
case DO_ONLY_CODE:
|
||||||
CopyTrailEntries();
|
if (CopyTrailEntries() < 0)
|
||||||
|
return -1;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
/* Check CRC */
|
/* Check CRC */
|
||||||
myread(splfild, my_end_msg, 256);
|
if (myread(splfild, my_end_msg, 256) < 0)
|
||||||
if (strcmp(end_msg,my_end_msg) != 0)
|
return -1;
|
||||||
Yap_Error(FATAL_ERROR,TermNil,"corrupt saved state (bad trailing CRC)");
|
if (strcmp(end_msg,my_end_msg) != 0) {
|
||||||
|
Yap_ErrorMessage = "bad trailing CRC in saved state";
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* restore some heap registers */
|
/* restore some heap registers */
|
||||||
@ -1278,10 +1414,22 @@ OpenRestore(char *inpf, char *YapLibDir, CELL *Astate, CELL *ATrail, CELL *AStac
|
|||||||
Yap_ErrorMessage = NULL;
|
Yap_ErrorMessage = NULL;
|
||||||
if (inpf == NULL)
|
if (inpf == NULL)
|
||||||
inpf = StartUpFile;
|
inpf = StartUpFile;
|
||||||
|
#if __simplescalar__
|
||||||
|
/* does not implement getcwd */
|
||||||
|
strncpy(Yap_FileNameBuf,yap_pwd,YAP_FILENAME_MAX);
|
||||||
|
#elif HAVE_GETCWD
|
||||||
|
if (getcwd (Yap_FileNameBuf, YAP_FILENAME_MAX) == NULL)
|
||||||
|
Yap_FileNameBuf[0] = '\0';
|
||||||
|
#else
|
||||||
|
if (getwd (Yap_FileNameBuf) == NULL)
|
||||||
|
Yap_FileNameBuf[0] = '\0';
|
||||||
|
#endif
|
||||||
|
strncat(Yap_FileNameBuf, "/", YAP_FILENAME_MAX);
|
||||||
|
strncat(Yap_FileNameBuf, inpf, YAP_FILENAME_MAX);
|
||||||
if (inpf != NULL && (splfild = open_file(inpf, O_RDONLY)) > 0) {
|
if (inpf != NULL && (splfild = open_file(inpf, O_RDONLY)) > 0) {
|
||||||
if ((mode = commit_to_saved_state(inpf,Astate,ATrail,AStack,AHeap)) != FAIL_RESTORE) {
|
if ((mode = commit_to_saved_state(inpf,Astate,ATrail,AStack,AHeap)) != FAIL_RESTORE) {
|
||||||
Yap_ErrorMessage = NULL;
|
Yap_ErrorMessage = NULL;
|
||||||
return(mode);
|
return mode;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!Yap_dir_separator(inpf[0]) && !Yap_volume_header(inpf)) {
|
if (!Yap_dir_separator(inpf[0]) && !Yap_volume_header(inpf)) {
|
||||||
@ -1294,7 +1442,7 @@ OpenRestore(char *inpf, char *YapLibDir, CELL *Astate, CELL *ATrail, CELL *AStac
|
|||||||
if ((splfild = open_file(Yap_FileNameBuf, O_RDONLY)) > 0) {
|
if ((splfild = open_file(Yap_FileNameBuf, O_RDONLY)) > 0) {
|
||||||
if ((mode = commit_to_saved_state(Yap_FileNameBuf,Astate,ATrail,AStack,AHeap)) != FAIL_RESTORE) {
|
if ((mode = commit_to_saved_state(Yap_FileNameBuf,Astate,ATrail,AStack,AHeap)) != FAIL_RESTORE) {
|
||||||
Yap_ErrorMessage = NULL;
|
Yap_ErrorMessage = NULL;
|
||||||
return(mode);
|
return mode;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1306,7 +1454,7 @@ OpenRestore(char *inpf, char *YapLibDir, CELL *Astate, CELL *ATrail, CELL *AStac
|
|||||||
if ((splfild = open_file(Yap_FileNameBuf, O_RDONLY)) > 0) {
|
if ((splfild = open_file(Yap_FileNameBuf, O_RDONLY)) > 0) {
|
||||||
if ((mode = commit_to_saved_state(Yap_FileNameBuf,Astate,ATrail,AStack,AHeap)) != FAIL_RESTORE) {
|
if ((mode = commit_to_saved_state(Yap_FileNameBuf,Astate,ATrail,AStack,AHeap)) != FAIL_RESTORE) {
|
||||||
Yap_ErrorMessage = NULL;
|
Yap_ErrorMessage = NULL;
|
||||||
return(mode);
|
return mode;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1324,12 +1472,10 @@ OpenRestore(char *inpf, char *YapLibDir, CELL *Astate, CELL *ATrail, CELL *AStac
|
|||||||
}
|
}
|
||||||
/* could not open file */
|
/* could not open file */
|
||||||
if (Yap_ErrorMessage == NULL) {
|
if (Yap_ErrorMessage == NULL) {
|
||||||
Yap_Error(SYSTEM_ERROR,TermNil,"could not open %s,",inpf);
|
strncpy(Yap_FileNameBuf, inpf, YAP_FILENAME_MAX);
|
||||||
} else {
|
do_system_error(PERMISSION_ERROR_OPEN_SOURCE_SINK,"could not open saved state");
|
||||||
Yap_Error(SYSTEM_ERROR, TermNil, Yap_ErrorMessage);
|
|
||||||
}
|
}
|
||||||
Yap_ErrorMessage = NULL;
|
return FAIL_RESTORE;
|
||||||
return(FAIL_RESTORE);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -1403,11 +1549,10 @@ Yap_SavedInfo(char *FileName, char *YapLibDir, CELL *ATrail, CELL *AStack, CELL
|
|||||||
int mode;
|
int mode;
|
||||||
|
|
||||||
mode = OpenRestore(FileName, YapLibDir, &MyState, &MyTrail, &MyStack, &MyHeap);
|
mode = OpenRestore(FileName, YapLibDir, &MyState, &MyTrail, &MyStack, &MyHeap);
|
||||||
close_file();
|
|
||||||
if (mode == FAIL_RESTORE) {
|
if (mode == FAIL_RESTORE) {
|
||||||
Yap_ErrorMessage = NULL;
|
return -1;
|
||||||
return(0);
|
|
||||||
}
|
}
|
||||||
|
close_file();
|
||||||
if (! *AHeap)
|
if (! *AHeap)
|
||||||
*AHeap = MyHeap / 1024;
|
*AHeap = MyHeap / 1024;
|
||||||
if (mode != DO_ONLY_CODE && *AStack)
|
if (mode != DO_ONLY_CODE && *AStack)
|
||||||
@ -1481,7 +1626,8 @@ Restore(char *s, char *lib_dir)
|
|||||||
return(FALSE);
|
return(FALSE);
|
||||||
Yap_ShutdownLoadForeign();
|
Yap_ShutdownLoadForeign();
|
||||||
in_limbo = TRUE;
|
in_limbo = TRUE;
|
||||||
get_coded(restore_mode, old_ops);
|
if (get_coded(restore_mode, old_ops) < 0)
|
||||||
|
return FAIL_RESTORE;
|
||||||
restore_regs(restore_mode);
|
restore_regs(restore_mode);
|
||||||
in_limbo = FALSE;
|
in_limbo = FALSE;
|
||||||
/*#endif*/
|
/*#endif*/
|
||||||
@ -1518,7 +1664,7 @@ Restore(char *s, char *lib_dir)
|
|||||||
if (which_save == 2) {
|
if (which_save == 2) {
|
||||||
Yap_unify(ARG2, MkIntTerm(0));
|
Yap_unify(ARG2, MkIntTerm(0));
|
||||||
}
|
}
|
||||||
return(restore_mode);
|
return restore_mode;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
|
15
C/stdpreds.c
15
C/stdpreds.c
@ -11,8 +11,12 @@
|
|||||||
* File: stdpreds.c *
|
* File: stdpreds.c *
|
||||||
* comments: General-purpose C implemented system predicates *
|
* comments: General-purpose C implemented system predicates *
|
||||||
* *
|
* *
|
||||||
* Last rev: $Date: 2005-03-01 22:25:09 $,$Author: vsc $ *
|
* Last rev: $Date: 2005-03-02 18:35:46 $,$Author: vsc $ *
|
||||||
* $Log: not supported by cvs2svn $
|
* $Log: not supported by cvs2svn $
|
||||||
|
* Revision 1.83 2005/03/01 22:25:09 vsc
|
||||||
|
* fix pruning bug
|
||||||
|
* make DL_MALLOC less enthusiastic about walking through buckets.
|
||||||
|
*
|
||||||
* Revision 1.82 2005/02/21 16:50:04 vsc
|
* Revision 1.82 2005/02/21 16:50:04 vsc
|
||||||
* amd64 fixes
|
* amd64 fixes
|
||||||
* library fixes
|
* library fixes
|
||||||
@ -1063,7 +1067,14 @@ p_name(void)
|
|||||||
}
|
}
|
||||||
if (IsAtomTerm(t) && AtomOfTerm(t) == AtomNil) {
|
if (IsAtomTerm(t) && AtomOfTerm(t) == AtomNil) {
|
||||||
if ((NewT = get_num(String)) == TermNil) {
|
if ((NewT = get_num(String)) == TermNil) {
|
||||||
NewT = MkAtomTerm(Yap_LookupAtom(String));
|
Atom at;
|
||||||
|
while ((at = Yap_LookupAtom(String)) == NIL) {
|
||||||
|
if (!Yap_growheap(FALSE, 0, NULL)) {
|
||||||
|
Yap_Error(OUT_OF_HEAP_ERROR, TermNil, Yap_ErrorMessage);
|
||||||
|
return(FALSE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
NewT = MkAtomTerm(at);
|
||||||
}
|
}
|
||||||
return Yap_unify_constant(ARG1, NewT);
|
return Yap_unify_constant(ARG1, NewT);
|
||||||
} else {
|
} else {
|
||||||
|
@ -287,14 +287,14 @@ parse_yap_arguments(int argc, char *argv[], YAP_init_args *iap)
|
|||||||
host = *++argv;
|
host = *++argv;
|
||||||
argc--;
|
argc--;
|
||||||
if (host == NULL || host[0] == '-')
|
if (host == NULL || host[0] == '-')
|
||||||
YAP_Error("sockets must receive host to connect to");
|
YAP_Error(0,0L,"sockets must receive host to connect to");
|
||||||
p1 = *++argv;
|
p1 = *++argv;
|
||||||
argc--;
|
argc--;
|
||||||
if (p1 == NULL || p1[0] == '-')
|
if (p1 == NULL || p1[0] == '-')
|
||||||
YAP_Error("sockets must receive port to connect to");
|
YAP_Error(0,0L,"sockets must receive port to connect to");
|
||||||
port = strtol(p1, &ptr, 10);
|
port = strtol(p1, &ptr, 10);
|
||||||
if (ptr == NULL || ptr[0] != '\0')
|
if (ptr == NULL || ptr[0] != '\0')
|
||||||
YAP_Error("port argument to socket must be a number");
|
YAP_Error(0,0L,"port argument to socket must be a number");
|
||||||
YAP_InitSocks(host,port);
|
YAP_InitSocks(host,port);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -446,6 +446,8 @@ init_standard_system(int argc, char *argv[], YAP_init_args *iap)
|
|||||||
iap->PrologShouldHandleInterrupts = TRUE;
|
iap->PrologShouldHandleInterrupts = TRUE;
|
||||||
iap->Argc = argc;
|
iap->Argc = argc;
|
||||||
iap->Argv = argv;
|
iap->Argv = argv;
|
||||||
|
iap->ErrorNo = 0;
|
||||||
|
iap->ErrorCause = NULL;
|
||||||
|
|
||||||
BootMode = parse_yap_arguments(argc,argv,iap);
|
BootMode = parse_yap_arguments(argc,argv,iap);
|
||||||
|
|
||||||
@ -466,10 +468,14 @@ init_standard_system(int argc, char *argv[], YAP_init_args *iap)
|
|||||||
/* init memory */
|
/* init memory */
|
||||||
if (BootMode == YAP_BOOT_FROM_PROLOG ||
|
if (BootMode == YAP_BOOT_FROM_PROLOG ||
|
||||||
BootMode == YAP_FULL_BOOT_FROM_PROLOG) {
|
BootMode == YAP_FULL_BOOT_FROM_PROLOG) {
|
||||||
YAP_Init(iap);
|
BootMode = YAP_Init(iap);
|
||||||
} else {
|
} else {
|
||||||
BootMode = YAP_Init(iap);
|
BootMode = YAP_Init(iap);
|
||||||
}
|
}
|
||||||
|
if (iap->ErrorNo) {
|
||||||
|
/* boot failed */
|
||||||
|
YAP_Error(iap->ErrorNo,0L,iap->ErrorCause);
|
||||||
|
}
|
||||||
return(BootMode);
|
return(BootMode);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -552,7 +558,7 @@ main (int argc, char **argv)
|
|||||||
YAP_init_args init_args;
|
YAP_init_args init_args;
|
||||||
|
|
||||||
BootMode = init_standard_system(argc, argv, &init_args);
|
BootMode = init_standard_system(argc, argv, &init_args);
|
||||||
if (BootMode == YAP_BOOT_FROM_SAVED_ERROR) {
|
if (BootMode == YAP_BOOT_ERROR) {
|
||||||
fprintf(stderr,"[ FATAL ERROR: could not find saved state ]\n");
|
fprintf(stderr,"[ FATAL ERROR: could not find saved state ]\n");
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
26
docs/yap.tex
26
docs/yap.tex
@ -14029,7 +14029,7 @@ program contains two facts for the procedure @t{b}:
|
|||||||
|
|
||||||
int
|
int
|
||||||
main(int argc, char *argv[]) @{
|
main(int argc, char *argv[]) @{
|
||||||
if (YAP_FastInit("saved_state") == YAP_BOOT_FROM_SAVED_ERROR)
|
if (YAP_FastInit("saved_state") == YAP_BOOT_ERROR)
|
||||||
exit(1);
|
exit(1);
|
||||||
if (YAP_RunGoal(YAP_MkAtomTerm(YAP_LookupAtom("do")))) @{
|
if (YAP_RunGoal(YAP_MkAtomTerm(YAP_LookupAtom("do")))) @{
|
||||||
printf("Success\n");
|
printf("Success\n");
|
||||||
@ -14091,14 +14091,19 @@ Initialize a copy of YAP from @var{SavedState}. The copy is
|
|||||||
monolithic and currently must be loaded at the same address where it was
|
monolithic and currently must be loaded at the same address where it was
|
||||||
saved. @code{YAP_FastInit} is a simpler version of @code{YAP_Init}.
|
saved. @code{YAP_FastInit} is a simpler version of @code{YAP_Init}.
|
||||||
|
|
||||||
@item YAP_Init(@code{char *} @var{SavedState}, @code{int}
|
@item YAP_Init(@var{InitInfo})
|
||||||
@var{HeapSize}, @code{int} @var{StackSize}, @code{int}
|
@findex YAP_Init/1
|
||||||
@var{TrailSize}, @code{int} @var{NumberofWorkers}, @code{int}
|
Initialize YAP. The arguments are in a @code{C}
|
||||||
|
structure of type @code{YAP_init_args}.
|
||||||
|
|
||||||
|
The fields of @var{InitInfo} are @code{char *} @var{SavedState},
|
||||||
|
@code{int} @var{HeapSize}, @code{int} @var{StackSize}, @code{int}
|
||||||
|
@var{TrailSize}, @code{int} @var{NumberofWorkers}, @code{int}
|
||||||
@var{SchedulerLoop}, @code{int} @var{DelayedReleaseLoad}, @code{int}
|
@var{SchedulerLoop}, @code{int} @var{DelayedReleaseLoad}, @code{int}
|
||||||
@var{argc}, @code{char **} @var{argv})
|
@var{argc}, @code{char **} @var{argv}, @code{int} @var{ErrorNo}, and
|
||||||
@findex YAP_Init/9
|
@code{char *} @var{ErrorCause}. The function returns an integer, which
|
||||||
Initialize YAP. In the future the arguments as a single @code{C}
|
indicates the current status. If the result is @code{YAP_BOOT_ERROR}
|
||||||
structure.
|
booting failed.
|
||||||
|
|
||||||
If @var{SavedState} is not NULL, try to open and restore the file
|
If @var{SavedState} is not NULL, try to open and restore the file
|
||||||
@var{SavedState}. Initially YAP will search in the current directory. If
|
@var{SavedState}. Initially YAP will search in the current directory. If
|
||||||
@ -14120,6 +14125,11 @@ The argument count @var{argc} and string of arguments @var{argv}
|
|||||||
arguments are to be passed to user programs as the arguments used to
|
arguments are to be passed to user programs as the arguments used to
|
||||||
call YAP.
|
call YAP.
|
||||||
|
|
||||||
|
If booting failed you may consult @code{ErrorNo} and @code{ErrorCause}
|
||||||
|
for the cause of the error, or call
|
||||||
|
@code{YAP_Error(ErrorNo,0L,ErrorCause)} to do default processing.
|
||||||
|
|
||||||
|
|
||||||
@item @code{void} YAP_PutValue(@code{Atom} @var{at}, @code{Term} @var{value})
|
@item @code{void} YAP_PutValue(@code{Atom} @var{at}, @code{Term} @var{value})
|
||||||
@findex YAP_PutValue/2
|
@findex YAP_PutValue/2
|
||||||
Associate the term @var{value} with the atom @var{at}. The term
|
Associate the term @var{value} with the atom @var{at}. The term
|
||||||
|
@ -24,6 +24,10 @@
|
|||||||
|
|
||||||
#include "yap_structs.h"
|
#include "yap_structs.h"
|
||||||
|
|
||||||
|
#if HAVE_STDARG_H
|
||||||
|
#include <stdarg.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifndef _yap_c_interface_h
|
#ifndef _yap_c_interface_h
|
||||||
|
|
||||||
#define _yap_c_interface_h 1
|
#define _yap_c_interface_h 1
|
||||||
@ -220,8 +224,8 @@ extern X_API YAP_Bool PROTO(YAP_GoalHasException,(YAP_Term *));
|
|||||||
/* int YAP_Reset(void) */
|
/* int YAP_Reset(void) */
|
||||||
extern X_API void PROTO(YAP_Reset,(void));
|
extern X_API void PROTO(YAP_Reset,(void));
|
||||||
|
|
||||||
/* void YAP_Error(char *) */
|
/* void YAP_Error(int, YAP_Term, char *,...) */
|
||||||
extern X_API void PROTO(YAP_Error,(char *));
|
extern X_API void PROTO(YAP_Error,(int, YAP_Term, char *, ...));
|
||||||
|
|
||||||
/* YAP_Term YAP_Read(int (*)(void)) */
|
/* YAP_Term YAP_Read(int (*)(void)) */
|
||||||
extern X_API YAP_Term PROTO(YAP_Read,(int (*)(void)));
|
extern X_API YAP_Term PROTO(YAP_Read,(int (*)(void)));
|
||||||
|
@ -53,7 +53,7 @@ typedef struct AtomEntry *YAP_Atom;
|
|||||||
#define YAP_BOOT_FROM_SAVED_CODE 1
|
#define YAP_BOOT_FROM_SAVED_CODE 1
|
||||||
#define YAP_BOOT_FROM_SAVED_STACKS 2
|
#define YAP_BOOT_FROM_SAVED_STACKS 2
|
||||||
#define YAP_FULL_BOOT_FROM_PROLOG 4
|
#define YAP_FULL_BOOT_FROM_PROLOG 4
|
||||||
#define YAP_BOOT_FROM_SAVED_ERROR -1
|
#define YAP_BOOT_ERROR -1
|
||||||
|
|
||||||
#define YAP_WRITE_QUOTED 0
|
#define YAP_WRITE_QUOTED 0
|
||||||
#define YAP_WRITE_HANDLE_VARS 1
|
#define YAP_WRITE_HANDLE_VARS 1
|
||||||
@ -98,6 +98,10 @@ typedef struct {
|
|||||||
int Argc;
|
int Argc;
|
||||||
/* array of arguments as seen by Prolog */
|
/* array of arguments as seen by Prolog */
|
||||||
char **Argv;
|
char **Argv;
|
||||||
|
/* errornumber */
|
||||||
|
int ErrorNo;
|
||||||
|
/* errorstring */
|
||||||
|
char *ErrorCause;
|
||||||
} YAP_init_args;
|
} YAP_init_args;
|
||||||
|
|
||||||
|
|
||||||
|
@ -10,7 +10,7 @@
|
|||||||
* File: Yap.h.m4 *
|
* File: Yap.h.m4 *
|
||||||
* mods: *
|
* mods: *
|
||||||
* comments: main header file for YAP *
|
* comments: main header file for YAP *
|
||||||
* version: $Id: Yap.h.m4,v 1.79 2005-02-08 18:38:02 vsc Exp $ *
|
* version: $Id: Yap.h.m4,v 1.80 2005-03-02 18:35:49 vsc Exp $ *
|
||||||
*************************************************************************/
|
*************************************************************************/
|
||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
@ -706,11 +706,11 @@ and RefOfTerm(t) : Term -> DBRef = ...
|
|||||||
|
|
||||||
extern ADDR Yap_HeapBase;
|
extern ADDR Yap_HeapBase;
|
||||||
|
|
||||||
#define MAX_ERROR_MSG_SIZE 256
|
|
||||||
|
|
||||||
/* This is ok for Linux, should be ok for everyone */
|
/* This is ok for Linux, should be ok for everyone */
|
||||||
#define YAP_FILENAME_MAX 1024
|
#define YAP_FILENAME_MAX 1024
|
||||||
|
|
||||||
|
#define MAX_ERROR_MSG_SIZE YAP_FILENAME_MAX
|
||||||
|
|
||||||
#ifdef THREADS
|
#ifdef THREADS
|
||||||
typedef struct thread_globs {
|
typedef struct thread_globs {
|
||||||
ADDR local_base;
|
ADDR local_base;
|
||||||
|
Reference in New Issue
Block a user