2015-06-18 01:20:05 +01:00
|
|
|
/*************************************************************************
|
2015-07-06 12:03:16 +01:00
|
|
|
* *
|
|
|
|
* YAP Prolog *
|
2015-07-23 01:33:30 +01:00
|
|
|
* *
|
2015-07-06 12:03:16 +01:00
|
|
|
* Yap Prolog was developed at NCCUP - Universidade do Porto *
|
|
|
|
* *
|
|
|
|
* Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997 *
|
|
|
|
* *
|
|
|
|
**************************************************************************
|
|
|
|
* *
|
|
|
|
* File: iopreds.c *
|
|
|
|
* Last rev: 5/2/88 *
|
|
|
|
* mods: *
|
|
|
|
* comments: Input/Output C implemented predicates *
|
|
|
|
* *
|
|
|
|
*************************************************************************/
|
2015-06-18 01:20:05 +01:00
|
|
|
#ifdef SCCS
|
|
|
|
static char SccsId[] = "%W% %G%";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
2015-07-06 12:03:16 +01:00
|
|
|
* This file includes the definition of a miscellania of standard predicates
|
|
|
|
* for yap refering to: Files and GLOBAL_Streams, Simple Input/Output,
|
|
|
|
*
|
|
|
|
*/
|
2015-06-18 01:20:05 +01:00
|
|
|
|
|
|
|
#include "Yap.h"
|
|
|
|
#include "Yatom.h"
|
|
|
|
#include "YapHeap.h"
|
2015-07-06 12:03:16 +01:00
|
|
|
#include "YapFlags.h"
|
2015-06-18 01:20:05 +01:00
|
|
|
#include "yapio.h"
|
|
|
|
#include "eval.h"
|
|
|
|
#include "YapText.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#if HAVE_STDARG_H
|
|
|
|
#include <stdarg.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_CTYPE_H
|
|
|
|
#include <ctype.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_WCTYPE_H
|
|
|
|
#include <wctype.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_SYS_TIME_H
|
|
|
|
#include <sys/time.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_SYS_TYPES_H
|
|
|
|
#include <sys/types.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SYS_STAT_H
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_SYS_SELECT_H && !_MSC_VER && !defined(__MINGW32__)
|
|
|
|
#include <sys/select.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_STRING_H
|
|
|
|
#include <string.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_SIGNAL_H
|
|
|
|
#include <signal.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_FCNTL_H
|
|
|
|
/* for O_BINARY and O_TEXT in WIN32 */
|
|
|
|
#include <fcntl.h>
|
|
|
|
#endif
|
|
|
|
#ifdef _WIN32
|
|
|
|
#if HAVE_IO_H
|
|
|
|
/* Windows */
|
|
|
|
#include <io.h>
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#if !HAVE_STRNCAT
|
2015-08-18 20:37:39 +01:00
|
|
|
#define strncat(X, Y, Z) strcat(X, Y)
|
2015-06-18 01:20:05 +01:00
|
|
|
#endif
|
|
|
|
#if !HAVE_STRNCPY
|
2015-08-18 20:37:39 +01:00
|
|
|
#define strncpy(X, Y, Z) strcpy(X, Y)
|
2015-06-18 01:20:05 +01:00
|
|
|
#endif
|
|
|
|
#if _MSC_VER || defined(__MINGW32__)
|
|
|
|
#if HAVE_SOCKET
|
|
|
|
#include <winsock2.h>
|
|
|
|
#endif
|
|
|
|
#include <windows.h>
|
|
|
|
#ifndef S_ISDIR
|
2015-08-18 20:37:39 +01:00
|
|
|
#define S_ISDIR(x) (((x)&_S_IFDIR) == _S_IFDIR)
|
2015-06-18 01:20:05 +01:00
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#include "iopreds.h"
|
|
|
|
|
|
|
|
#if _MSC_VER || defined(__MINGW32__)
|
|
|
|
#define SYSTEM_STAT _stat
|
|
|
|
#else
|
|
|
|
#define SYSTEM_STAT stat
|
|
|
|
#endif
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static void clean_vars(VarEntry *p) {
|
|
|
|
if (p == NULL)
|
|
|
|
return;
|
2015-06-18 01:20:05 +01:00
|
|
|
p->VarAdr = TermNil;
|
|
|
|
clean_vars(p->VarLeft);
|
|
|
|
clean_vars(p->VarRight);
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef PAR
|
|
|
|
|
|
|
|
#ifdef O_QUASIQUOTATIONS
|
|
|
|
/** '$qq_open'(+QQRange, -Stream) is det.
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
Opens a quasi-quoted memory range.
|
|
|
|
|
|
|
|
@arg QQRange is a term '$quasi_quotation'(ReadData, Start, Length)
|
|
|
|
@arg Stream is a UTF-8 encoded string, whose position indication
|
|
|
|
reflects the location in the real file.
|
2015-06-18 01:20:05 +01:00
|
|
|
*/
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int qq_open(USES_REGS1) {
|
|
|
|
PRED_LD
|
|
|
|
|
|
|
|
Term t = Deref(ARG1);
|
|
|
|
if (!IsVarTerm(t) && IsApplTerm(t) && FunctorOfTerm(t) =
|
|
|
|
FunctorDQuasiQuotation) {
|
|
|
|
void *ptr;
|
|
|
|
char *start;
|
|
|
|
size_t l int s;
|
|
|
|
Term t0, t1, t2;
|
|
|
|
|
|
|
|
if (IsPointerTerm((t0 = ArgOfTerm(1, t))) &&
|
|
|
|
IsPointerTerm((t1 = ArgOfTerm(2, t))) &&
|
|
|
|
IsIntegerTerm((t2 = ArgOfTerm(3, t)))) {
|
|
|
|
ptr = PointerOfTerm(t0);
|
|
|
|
start = PointerOfTerm(t1);
|
|
|
|
len = IntegerOfTerm(t2);
|
|
|
|
if ((s = Yap_open_buf_read_stream(start, len, ENC_UTF8, MEM_BUF_USER)) <
|
|
|
|
0)
|
|
|
|
return false;
|
|
|
|
return Yap_unify(ARG2, Yap_MkStream(s));
|
|
|
|
} else {
|
|
|
|
Yap_Error(TYPE_ERROR_READ_CONTEXT, t);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static int parse_quasi_quotations(ReadData _PL_rd ARG_LD) {
|
|
|
|
if (_PL_rd->qq_tail) {
|
|
|
|
term_t av;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (!PL_unify_nil(_PL_rd->qq_tail))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!_PL_rd->quasi_quotations) {
|
|
|
|
if ((av = PL_new_term_refs(2)) && PL_put_term(av + 0, _PL_rd->qq) &&
|
|
|
|
#if __YAP_PROLOG__
|
|
|
|
PL_put_atom(av + 1, YAP_SWIAtomFromAtom(_PL_rd->module->AtomOfME)) &&
|
|
|
|
#else
|
|
|
|
PL_put_atom(av + 1, _PL_rd->module->name) &&
|
|
|
|
#endif
|
|
|
|
PL_cons_functor_v(av, FUNCTOR_dparse_quasi_quotations2, av)) {
|
|
|
|
term_t ex;
|
|
|
|
rc = callProlog(MODULE_system, av + 0, PL_Q_CATCH_EXCEPTION, &ex);
|
|
|
|
if (rc)
|
|
|
|
return TRUE;
|
|
|
|
_PL_rd->exception = ex;
|
|
|
|
_PL_rd->has_exception = TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
2015-06-18 01:20:05 +01:00
|
|
|
} else
|
2015-08-18 20:37:39 +01:00
|
|
|
return TRUE;
|
|
|
|
} else if (_PL_rd->quasi_quotations) /* user option, but no quotes */
|
|
|
|
{
|
|
|
|
return PL_unify_nil(_PL_rd->quasi_quotations);
|
|
|
|
} else
|
2015-06-18 01:20:05 +01:00
|
|
|
return TRUE;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
#endif /*O_QUASIQUOTATIONS*/
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
#define READ_DEFS() \
|
|
|
|
PAR("comments", filler, READ_COMMENTS), PAR("module", isatom, READ_MODULE), \
|
|
|
|
PAR("priority", nat, READ_PRIORITY), \
|
|
|
|
PAR("quasi_quotations", filler, READ_QUASI_QUOTATIONS), \
|
|
|
|
PAR("term_position", filler, READ_TERM_POSITION), \
|
|
|
|
PAR("syntax_errors", isatom, READ_SYNTAX_ERRORS), \
|
|
|
|
PAR("singletons", filler, READ_SINGLETONS), \
|
|
|
|
PAR("variables", filler, READ_VARIABLES), \
|
|
|
|
PAR("variable_names", filler, READ_VARIABLE_NAMES), \
|
|
|
|
PAR("character_escapes", boolean, READ_CHARACTER_ESCAPES), \
|
|
|
|
PAR("backquoted_string", isatom, READ_BACKQUOTED_STRING), \
|
|
|
|
PAR("cycles", ok, READ_CYCLES), PAR(NULL, ok, READ_END)
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
#define PAR(x, y, z) z
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
typedef enum open_enum_choices { READ_DEFS() } read_choices_t;
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
#undef PAR
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
#define PAR(x, y, z) \
|
|
|
|
{ x, y, z }
|
|
|
|
|
|
|
|
static const param_t read_defs[] = {READ_DEFS()};
|
|
|
|
#undef PAR
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Syntax Error Handler
|
|
|
|
*
|
|
|
|
* @par tokptr: the sequence of tokens
|
|
|
|
* @par sno: the stream numbet
|
|
|
|
*
|
|
|
|
* Implicit arguments:
|
|
|
|
* +
|
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
Term Yap_syntax_error(TokEntry *errtok, int sno) {
|
2015-06-18 01:20:05 +01:00
|
|
|
CACHE_REGS
|
2015-08-18 20:37:39 +01:00
|
|
|
Term info;
|
2015-07-06 12:03:16 +01:00
|
|
|
Term startline, errline, endline;
|
2015-08-18 20:37:39 +01:00
|
|
|
Term tf[4];
|
2015-09-21 23:05:36 +01:00
|
|
|
Term *tailp = tf + 3;
|
2015-06-18 01:20:05 +01:00
|
|
|
CELL *Hi = HR;
|
2015-08-18 20:37:39 +01:00
|
|
|
TokEntry *tok = LOCAL_tokptr;
|
2015-07-28 04:22:44 +01:00
|
|
|
Int cline = tok->TokPos;
|
2015-08-18 20:37:39 +01:00
|
|
|
|
|
|
|
*tailp = TermNl;
|
2015-07-28 04:22:44 +01:00
|
|
|
startline = MkIntegerTerm(cline);
|
2015-06-18 01:20:05 +01:00
|
|
|
clean_vars(LOCAL_VarTable);
|
|
|
|
clean_vars(LOCAL_AnonVarTable);
|
2015-08-18 20:37:39 +01:00
|
|
|
if (errtok != LOCAL_toktide) {
|
|
|
|
errtok = LOCAL_toktide;
|
|
|
|
}
|
|
|
|
errline = MkIntegerTerm(errtok->TokPos);
|
|
|
|
while (tok) {
|
|
|
|
Term ts[2];
|
|
|
|
|
|
|
|
if (HR > ASP - 1024) {
|
|
|
|
tf[3] = TermNil;
|
|
|
|
errline = MkIntegerTerm(0);
|
|
|
|
endline = MkIntegerTerm(0);
|
|
|
|
/* for some reason moving this earlier confuses gcc on solaris */
|
|
|
|
HR = Hi;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (tok->TokPos != cline) {
|
|
|
|
*tailp = MkPairTerm(TermNewLine, TermNil);
|
|
|
|
tailp = RepPair(*tailp) + 1;
|
|
|
|
cline = tok->TokPos;
|
|
|
|
}
|
|
|
|
if (tok == errtok && tok->Tok != Error_tok) {
|
|
|
|
*tailp = MkPairTerm(MkAtomTerm(AtomError), TermNil);
|
|
|
|
tailp = RepPair(*tailp) + 1;
|
|
|
|
}
|
|
|
|
info = tok->TokInfo;
|
|
|
|
switch (tok->Tok) {
|
|
|
|
case Name_tok: {
|
|
|
|
Term t0[1];
|
|
|
|
t0[0] = MkAtomTerm((Atom)info);
|
|
|
|
ts[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomAtom, 1), 1, t0);
|
|
|
|
} break;
|
2015-09-21 23:05:36 +01:00
|
|
|
case QuasiQuotes_tok:
|
|
|
|
{
|
|
|
|
Term t0[2];
|
|
|
|
t0[0] = MkAtomTerm(Yap_LookupAtom("<QQ>"));
|
|
|
|
ts[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomAtom, 1), 1, t0);
|
|
|
|
} break;
|
|
|
|
case WQuasiQuotes_tok:
|
|
|
|
{
|
|
|
|
Term t0[2];
|
|
|
|
t0[0] = MkAtomTerm(Yap_LookupAtom("<WideQQ>"));
|
|
|
|
ts[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomAtom, 1), 1, t0);
|
|
|
|
} break;
|
|
|
|
case Number_tok:
|
2015-08-18 20:37:39 +01:00
|
|
|
ts[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomNumber, 1), 1, &(tok->TokInfo));
|
|
|
|
break;
|
|
|
|
case Var_tok: {
|
|
|
|
Term t[3];
|
|
|
|
VarEntry *varinfo = (VarEntry *)info;
|
|
|
|
|
|
|
|
t[0] = MkIntTerm(0);
|
2015-09-21 23:05:36 +01:00
|
|
|
t[1] = Yap_CharsToString(varinfo->VarRep, ENC_ISO_LATIN1 PASS_REGS);
|
2015-08-18 20:37:39 +01:00
|
|
|
if (varinfo->VarAdr == TermNil) {
|
|
|
|
t[2] = varinfo->VarAdr = MkVarTerm();
|
|
|
|
} else {
|
|
|
|
t[2] = varinfo->VarAdr;
|
2015-07-28 04:22:44 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
ts[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomGVar, 3), 3, t);
|
|
|
|
} break;
|
|
|
|
case String_tok: {
|
2015-09-21 23:05:36 +01:00
|
|
|
Term t0 = Yap_CharsToTDQ((char *)info, CurrentModule, ENC_ISO_LATIN1 PASS_REGS);
|
2015-08-18 20:37:39 +01:00
|
|
|
ts[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomString, 1), 1, &t0);
|
|
|
|
} break;
|
|
|
|
case WString_tok: {
|
|
|
|
Term t0 = Yap_WCharsToTDQ((wchar_t *)info, CurrentModule PASS_REGS);
|
|
|
|
ts[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomString, 1), 1, &t0);
|
|
|
|
} break;
|
|
|
|
case BQString_tok: {
|
2015-09-21 23:05:36 +01:00
|
|
|
Term t0 = Yap_CharsToTBQ((char *)info, CurrentModule, ENC_ISO_LATIN1 PASS_REGS);
|
2015-08-18 20:37:39 +01:00
|
|
|
ts[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomString, 1), 1, &t0);
|
|
|
|
} break;
|
|
|
|
case WBQString_tok: {
|
|
|
|
Term t0 = Yap_WCharsToTBQ((wchar_t *)info, CurrentModule PASS_REGS);
|
|
|
|
ts[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomString, 1), 1, &t0);
|
|
|
|
} break;
|
|
|
|
case Error_tok: {
|
|
|
|
ts[0] = MkAtomTerm(AtomError);
|
|
|
|
} break;
|
|
|
|
case eot_tok:
|
|
|
|
endline = MkIntegerTerm(tok->TokPos);
|
|
|
|
ts[0] = MkAtomTerm(Yap_LookupAtom("EOT"));
|
|
|
|
|
|
|
|
break;
|
|
|
|
case Ponctuation_tok: {
|
|
|
|
char s[2];
|
|
|
|
s[1] = '\0';
|
|
|
|
if ((info) == 'l') {
|
|
|
|
s[0] = '(';
|
|
|
|
} else {
|
|
|
|
s[0] = (char)info;
|
2015-07-28 04:22:44 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
ts[0] = MkAtomTerm(Yap_LookupAtom(s));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tok = tok->TokNext;
|
|
|
|
if (!tok)
|
|
|
|
break;
|
|
|
|
*tailp = MkPairTerm(ts[0], TermNil);
|
|
|
|
tailp = RepPair(*tailp) + 1;
|
2015-07-28 04:22:44 +01:00
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
{
|
|
|
|
Term t[3];
|
2015-07-06 12:03:16 +01:00
|
|
|
t[0] = startline;
|
|
|
|
t[1] = errline;
|
|
|
|
t[2] = endline;
|
2015-08-18 20:37:39 +01:00
|
|
|
tf[1] = Yap_MkApplTerm(Yap_MkFunctor(AtomBetween, 3), 3, t);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-28 04:22:44 +01:00
|
|
|
/* 0: id */
|
|
|
|
/* 1: strat, error, end line */
|
|
|
|
/*2 msg */
|
|
|
|
if (LOCAL_ErrorMessage)
|
2015-08-18 20:37:39 +01:00
|
|
|
tf[0] = MkStringTerm(LOCAL_ErrorMessage);
|
2015-07-28 04:22:44 +01:00
|
|
|
else
|
2015-08-18 20:37:39 +01:00
|
|
|
tf[0] = MkStringTerm("");
|
2015-07-28 04:22:44 +01:00
|
|
|
/* file */
|
2015-08-18 20:37:39 +01:00
|
|
|
tf[2] = Yap_StreamUserName(sno);
|
|
|
|
tf[1] = Yap_MkApplTerm(FunctorSyntaxError, 4, tf);
|
2015-09-21 23:05:36 +01:00
|
|
|
tf[0] = MkAtomTerm(AtomSyntaxError);
|
2015-08-18 20:37:39 +01:00
|
|
|
return Yap_MkApplTerm(FunctorError, 2, tf);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
typedef struct FEnv {
|
2015-06-18 01:20:05 +01:00
|
|
|
Term qq, tp, sp, np, vp, ce;
|
2015-08-18 20:37:39 +01:00
|
|
|
Term tpos; /// initial position of the term to be read.
|
|
|
|
Term t; /// the output term
|
|
|
|
TokEntry *tokstart; /// the token list
|
|
|
|
TokEntry *toklast; /// the last token
|
|
|
|
CELL *old_H; /// initial H, will be reset on stack overflow.
|
|
|
|
tr_fr_ptr old_TR; /// initial TR
|
|
|
|
xarg *args; /// input args
|
|
|
|
bool reading_clause; /// read_clause
|
|
|
|
size_t nargs; /// arity of current procedure
|
2015-06-18 01:20:05 +01:00
|
|
|
} FEnv;
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
typedef struct renv {
|
|
|
|
Term cm, bq;
|
2015-08-18 20:37:39 +01:00
|
|
|
bool ce, sw;
|
2015-07-06 12:03:16 +01:00
|
|
|
Term sy;
|
2015-06-18 01:20:05 +01:00
|
|
|
UInt cpos;
|
|
|
|
#if HAVE_FGETPOS
|
|
|
|
fpos_t rpos;
|
|
|
|
#endif
|
2015-07-06 12:03:16 +01:00
|
|
|
int prio;
|
2015-06-18 01:20:05 +01:00
|
|
|
int ungetc_oldc;
|
|
|
|
int had_ungetc;
|
|
|
|
bool seekable;
|
|
|
|
} REnv;
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static xarg *setClauseReadEnv(Term opts, FEnv *fe, struct renv *re,
|
|
|
|
int inp_stream);
|
|
|
|
static xarg *setReadEnv(Term opts, FEnv *fe, struct renv *re, int inp_stream) {
|
2015-06-18 01:20:05 +01:00
|
|
|
CACHE_REGS
|
2015-08-18 20:37:39 +01:00
|
|
|
re->cm = CurrentModule;
|
|
|
|
xarg *args = Yap_ArgListToVector(opts, read_defs, READ_END);
|
2015-07-06 12:03:16 +01:00
|
|
|
if (args == NULL) {
|
2015-08-18 20:37:39 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
re->bq = getBackQuotesFlag();
|
2015-06-18 01:20:05 +01:00
|
|
|
if (args[READ_MODULE].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
CurrentModule = args[READ_MODULE].tvalue;
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
if (args[READ_BACKQUOTED_STRING].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
if (!setBackQuotesFlag(args[READ_BACKQUOTED_STRING].tvalue))
|
|
|
|
return false;
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
if (args[READ_QUASI_QUOTATIONS].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->qq = args[READ_QUASI_QUOTATIONS].tvalue;
|
|
|
|
} else {
|
|
|
|
fe->qq = 0;
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
if (args[READ_TERM_POSITION].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->tp = args[READ_TERM_POSITION].tvalue;
|
|
|
|
} else {
|
|
|
|
fe->tp = 0;
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
if (args[READ_SINGLETONS].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->sp = args[READ_SINGLETONS].tvalue;
|
|
|
|
} else if (args[READ_SINGLETONS].used) {
|
|
|
|
fe->sp = MkVarTerm();
|
|
|
|
} else {
|
|
|
|
fe->sp = 0;
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
if (args[READ_SYNTAX_ERRORS].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
re->sy = args[READ_SYNTAX_ERRORS].tvalue;
|
|
|
|
} else {
|
2015-07-06 12:03:16 +01:00
|
|
|
re->sy = TermError; // getYapFlag( MkAtomTerm(AtomSyntaxErrors) );
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
if (args[READ_VARIABLES].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->vp = args[READ_VARIABLES].tvalue;
|
|
|
|
} else {
|
|
|
|
fe->vp = 0;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
if (args[READ_VARIABLE_NAMES].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->np = args[READ_VARIABLE_NAMES].tvalue;
|
|
|
|
} else {
|
|
|
|
fe->np = 0;
|
|
|
|
}
|
|
|
|
if (args[READ_CHARACTER_ESCAPES].used ||
|
|
|
|
Yap_CharacterEscapes(CurrentModule)) {
|
|
|
|
fe->ce = true;
|
|
|
|
} else {
|
|
|
|
fe->ce = false;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
re->seekable = (GLOBAL_Stream[inp_stream].status & Seekable_Stream_f) != 0;
|
|
|
|
if (re->seekable) {
|
2015-08-18 20:37:39 +01:00
|
|
|
if (GLOBAL_Stream[inp_stream].stream_getc == PlUnGetc) {
|
|
|
|
re->had_ungetc = TRUE;
|
|
|
|
re->ungetc_oldc = GLOBAL_Stream[inp_stream].och;
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
#if HAVE_FGETPOS
|
2015-08-18 20:37:39 +01:00
|
|
|
fgetpos(GLOBAL_Stream[inp_stream].file, &re->rpos);
|
2015-06-18 01:20:05 +01:00
|
|
|
#else
|
2015-08-18 20:37:39 +01:00
|
|
|
re->cpos = GLOBAL_Stream[inp_stream].charcount;
|
2015-06-18 01:20:05 +01:00
|
|
|
#endif
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
if (args[READ_PRIORITY].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
re->prio = IntegerOfTerm(args[READ_PRIORITY].tvalue);
|
|
|
|
if (re->prio > 1200) {
|
|
|
|
Yap_Error(DOMAIN_ERROR_OPERATOR_PRIORITY, opts,
|
|
|
|
"max priority in Prolog is 1200, not %ld", re->prio);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
} else {
|
|
|
|
re->prio = LOCAL_default_priority;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
return args;
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
typedef enum {
|
2015-08-18 20:37:39 +01:00
|
|
|
YAP_START_PARSING, /// initialization
|
2015-07-06 12:03:16 +01:00
|
|
|
YAP_SCANNING, /// input to list of tokens
|
2015-08-18 20:37:39 +01:00
|
|
|
YAP_SCANNING_ERROR, /// serious error (eg oom); trying error handling, followd
|
|
|
|
/// by either restart or failure
|
|
|
|
YAP_PARSING, /// list of tokens to term
|
|
|
|
YAP_PARSING_ERROR, /// oom or syntax error
|
2015-07-06 12:03:16 +01:00
|
|
|
YAP_PARSING_FINISHED /// exit parser
|
|
|
|
} parser_state_t;
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
Int Yap_FirstLineInParse(void) {
|
2015-06-18 01:20:05 +01:00
|
|
|
CACHE_REGS
|
2015-08-18 20:37:39 +01:00
|
|
|
return LOCAL_StartLineCount;
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-09-21 23:05:36 +01:00
|
|
|
#define PUSHFET(X) *HR++ = fe->X
|
|
|
|
#define POPFET(X) fe->X = *--HR
|
2015-08-18 20:37:39 +01:00
|
|
|
|
|
|
|
static void reset_regs(TokEntry *tokstart, FEnv *fe) {
|
2015-06-18 01:20:05 +01:00
|
|
|
CACHE_REGS
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
restore_machine_regs();
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
/* restart global */
|
2015-09-21 23:05:36 +01:00
|
|
|
PUSHFET(qq);
|
|
|
|
PUSHFET(tp);
|
|
|
|
PUSHFET(sp);
|
|
|
|
PUSHFET(np);
|
|
|
|
PUSHFET(vp);
|
|
|
|
PUSHFET(tpos);
|
|
|
|
PUSHFET(t);
|
2015-07-06 12:03:16 +01:00
|
|
|
HR = fe->old_H;
|
2015-06-18 01:20:05 +01:00
|
|
|
TR = (tr_fr_ptr)LOCAL_ScannerStack;
|
2015-08-18 20:37:39 +01:00
|
|
|
LOCAL_Error_TYPE = YAP_NO_ERROR;
|
2015-07-06 12:03:16 +01:00
|
|
|
Yap_growstack_in_parser(&fe->old_TR, &tokstart, &LOCAL_VarTable);
|
2015-06-18 01:20:05 +01:00
|
|
|
LOCAL_ScannerStack = (char *)TR;
|
2015-07-06 12:03:16 +01:00
|
|
|
TR = fe->old_TR;
|
2015-09-21 23:05:36 +01:00
|
|
|
POPFET(t);
|
|
|
|
POPFET(tpos);
|
|
|
|
POPFET(vp);
|
|
|
|
POPFET(np);
|
|
|
|
POPFET(sp);
|
|
|
|
POPFET(tp);
|
|
|
|
POPFET(qq);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static bool complete_clause_processing(FEnv *fe, TokEntry *tokstarts, Term t);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static bool complete_processing(FEnv *fe, TokEntry *tokstart) {
|
2015-06-18 01:20:05 +01:00
|
|
|
CACHE_REGS
|
2015-09-21 23:05:36 +01:00
|
|
|
Term v1, v2, v3;
|
2015-06-18 01:20:05 +01:00
|
|
|
if (fe->vp) {
|
2015-08-18 20:37:39 +01:00
|
|
|
while (TRUE) {
|
|
|
|
fe->old_H = HR;
|
|
|
|
|
|
|
|
if (setjmp(LOCAL_IOBotch) == 0) {
|
|
|
|
if ((v1 = Yap_Variables(LOCAL_VarTable, TermNil)))
|
2015-09-21 23:05:36 +01:00
|
|
|
break;
|
2015-08-18 20:37:39 +01:00
|
|
|
} else {
|
|
|
|
reset_regs(tokstart, fe);
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
if (fe->np) {
|
2015-08-18 20:37:39 +01:00
|
|
|
while (true) {
|
|
|
|
fe->old_H = HR;
|
|
|
|
|
|
|
|
if (setjmp(LOCAL_IOBotch) == 0) {
|
2015-09-21 23:05:36 +01:00
|
|
|
if ((v2 = Yap_VarNames(LOCAL_VarTable, TermNil))) {
|
|
|
|
fe->old_H = HR;
|
|
|
|
break;
|
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
} else {
|
|
|
|
reset_regs(tokstart, fe);
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-23 01:33:30 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
if (fe->sp) {
|
|
|
|
while (TRUE) {
|
|
|
|
fe->old_H = HR;
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
if (setjmp(LOCAL_IOBotch) == 0) {
|
|
|
|
if ((v3 = Yap_Singletons(LOCAL_VarTable, TermNil)))
|
2015-09-21 23:05:36 +01:00
|
|
|
break;
|
2015-08-18 20:37:39 +01:00
|
|
|
} else {
|
|
|
|
reset_regs(tokstart, fe);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-09-21 23:05:36 +01:00
|
|
|
Yap_clean_tokenizer(tokstart, LOCAL_VarTable, LOCAL_AnonVarTable);
|
|
|
|
|
|
|
|
// trail must be ok by now.]
|
|
|
|
if ((!fe->vp || Yap_unify(v1, fe->vp)) &&
|
|
|
|
(!fe->np || Yap_unify(v2, fe->np)) && (!fe->tp || Yap_unify(v3, fe->tp)))
|
|
|
|
return fe->t;
|
|
|
|
return 0;
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t initParser(Term opts, FEnv *fe, REnv *re, int inp_stream,
|
|
|
|
int nargs);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t parse(REnv *re, FEnv *fe, int inp_stream);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t scanError(REnv *re, FEnv *fe, int inp_stream);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t scanEOF(FEnv *fe, int inp_stream);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t scan(REnv *re, FEnv *fe, int inp_stream);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t scanEOF(FEnv *fe, int inp_stream) {
|
2015-06-18 01:20:05 +01:00
|
|
|
CACHE_REGS
|
2015-08-18 20:37:39 +01:00
|
|
|
// bool store_comments = false;
|
|
|
|
TokEntry *tokstart = LOCAL_tokptr;
|
|
|
|
// check for an user abort
|
|
|
|
if (tokstart != NULL && tokstart->Tok != Ord(eot_tok)) {
|
|
|
|
/* we got the end of file from an abort */
|
|
|
|
if (LOCAL_ErrorMessage && !strcmp(LOCAL_ErrorMessage, "Abort")) {
|
|
|
|
fe->t = 0L;
|
|
|
|
Yap_clean_tokenizer(tokstart, LOCAL_VarTable, LOCAL_AnonVarTable);
|
|
|
|
return YAP_PARSING_FINISHED;
|
|
|
|
}
|
|
|
|
// a :- <eof>
|
|
|
|
/* we need to force the next read to also give end of file.*/
|
|
|
|
GLOBAL_Stream[inp_stream].status |= Push_Eof_Stream_f;
|
|
|
|
LOCAL_ErrorMessage = "end of file found before end of term";
|
|
|
|
return YAP_PARSING;
|
|
|
|
} else {
|
|
|
|
// <eof>
|
|
|
|
// return end_of_file
|
|
|
|
TR = (tr_fr_ptr)tokstart;
|
|
|
|
Yap_clean_tokenizer(tokstart, LOCAL_VarTable, LOCAL_AnonVarTable);
|
|
|
|
fe->t = MkAtomTerm(AtomEof);
|
|
|
|
if (fe->np && !Yap_unify(TermNil, fe->np))
|
|
|
|
fe->t = 0;
|
|
|
|
if (fe->sp && !Yap_unify(TermNil, fe->sp))
|
|
|
|
fe->t = 0;
|
|
|
|
if (fe->vp && !Yap_unify(TermNil, fe->vp))
|
|
|
|
fe->t = 0;
|
|
|
|
if (fe->tp && !Yap_unify(fe->tp, fe->tpos))
|
|
|
|
fe->t = 0;
|
|
|
|
post_process_eof(GLOBAL_Stream + inp_stream);
|
2015-07-06 12:03:16 +01:00
|
|
|
#if DEBUG
|
2015-08-18 20:37:39 +01:00
|
|
|
if (GLOBAL_Option['p' - 'a' + 1]) {
|
2015-07-23 01:33:30 +01:00
|
|
|
fprintf(stderr, "[ end_of_file %p ]\n", GLOBAL_Stream[inp_stream].name);
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
#endif
|
2015-08-18 20:37:39 +01:00
|
|
|
return YAP_PARSING_FINISHED;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t initParser(Term opts, FEnv *fe, REnv *re, int inp_stream,
|
|
|
|
int nargs) {
|
2015-07-06 12:03:16 +01:00
|
|
|
CACHE_REGS
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->old_H = HR;
|
2015-07-06 12:03:16 +01:00
|
|
|
LOCAL_ErrorMessage = NULL;
|
|
|
|
fe->old_TR = TR;
|
2015-06-18 01:20:05 +01:00
|
|
|
LOCAL_Error_TYPE = YAP_NO_ERROR;
|
|
|
|
LOCAL_SourceFileName = GLOBAL_Stream[inp_stream].name;
|
2015-07-06 12:03:16 +01:00
|
|
|
LOCAL_eot_before_eof = false;
|
|
|
|
fe->tpos = StreamPosition(inp_stream);
|
|
|
|
fe->reading_clause = nargs < 0;
|
|
|
|
if (fe->reading_clause) {
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->nargs = -nargs;
|
|
|
|
fe->args = setClauseReadEnv(opts, fe, re, inp_stream);
|
|
|
|
} else {
|
|
|
|
fe->nargs = nargs;
|
|
|
|
fe->args = setReadEnv(opts, fe, re, inp_stream);
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
if (GLOBAL_Stream[inp_stream].status & Push_Eof_Stream_f) {
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->t = MkAtomTerm(AtomEof);
|
|
|
|
GLOBAL_Stream[inp_stream].status &= ~Push_Eof_Stream_f;
|
|
|
|
return YAP_PARSING_FINISHED;
|
|
|
|
}
|
|
|
|
if (!fe->args) {
|
2015-07-06 12:03:16 +01:00
|
|
|
return YAP_PARSING_FINISHED;
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
return YAP_SCANNING;
|
|
|
|
}
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t scan(REnv *re, FEnv *fe, int inp_stream) {
|
2015-07-06 12:03:16 +01:00
|
|
|
CACHE_REGS
|
|
|
|
/* preserve value of H after scanning: otherwise we may lose strings
|
|
|
|
and floats */
|
|
|
|
LOCAL_tokptr = LOCAL_toktide =
|
2015-08-18 20:37:39 +01:00
|
|
|
Yap_tokenizer(GLOBAL_Stream + inp_stream, false, &fe->tpos);
|
2015-07-06 12:03:16 +01:00
|
|
|
if (LOCAL_ErrorMessage)
|
|
|
|
return YAP_SCANNING_ERROR;
|
2015-08-18 20:37:39 +01:00
|
|
|
if (LOCAL_tokptr->Tok != Ord(eot_tok)) {
|
|
|
|
// next step
|
|
|
|
return YAP_PARSING;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
return scanEOF(fe, inp_stream);
|
|
|
|
}
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t scanError(REnv *re, FEnv *fe, int inp_stream) {
|
2015-07-06 12:03:16 +01:00
|
|
|
CACHE_REGS
|
|
|
|
fe->t = 0;
|
|
|
|
// running out of memory
|
2015-09-25 10:57:26 +01:00
|
|
|
if (LOCAL_Error_TYPE == RESOURCE_ERROR_TRAIL) {
|
2015-08-18 20:37:39 +01:00
|
|
|
LOCAL_Error_TYPE = YAP_NO_ERROR;
|
|
|
|
if (!Yap_growtrail(sizeof(CELL) * K16, FALSE)) {
|
|
|
|
return YAP_PARSING_FINISHED;
|
|
|
|
}
|
2015-09-25 10:57:26 +01:00
|
|
|
} else if (LOCAL_Error_TYPE == RESOURCE_ERROR_AUXILIARY_STACK) {
|
2015-08-18 20:37:39 +01:00
|
|
|
LOCAL_Error_TYPE = YAP_NO_ERROR;
|
|
|
|
if (!Yap_ExpandPreAllocCodeSpace(0, NULL, TRUE)) {
|
|
|
|
return YAP_PARSING_FINISHED;
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-09-25 10:57:26 +01:00
|
|
|
} else if (LOCAL_Error_TYPE == RESOURCE_ERROR_HEAP) {
|
2015-08-18 20:37:39 +01:00
|
|
|
LOCAL_Error_TYPE = YAP_NO_ERROR;
|
|
|
|
if (!Yap_growheap(FALSE, 0, NULL)) {
|
|
|
|
return YAP_PARSING_FINISHED;
|
|
|
|
}
|
2015-09-25 10:57:26 +01:00
|
|
|
} else if (LOCAL_Error_TYPE == RESOURCE_ERROR_STACK) {
|
2015-08-18 20:37:39 +01:00
|
|
|
LOCAL_Error_TYPE = YAP_NO_ERROR;
|
|
|
|
if (!Yap_gcl(LOCAL_Error_Size, fe->nargs, ENV, CP)) {
|
|
|
|
return YAP_PARSING_FINISHED;
|
|
|
|
}
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
// go back to the start
|
|
|
|
if (re->had_ungetc) {
|
2015-08-18 20:37:39 +01:00
|
|
|
GLOBAL_Stream[inp_stream].stream_getc = PlUnGetc;
|
|
|
|
GLOBAL_Stream[inp_stream].och = re->ungetc_oldc;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
if (re->seekable) {
|
2015-08-18 20:37:39 +01:00
|
|
|
if (GLOBAL_Stream[inp_stream].status & InMemory_Stream_f) {
|
|
|
|
GLOBAL_Stream[inp_stream].u.mem_string.pos = re->cpos;
|
|
|
|
} else if (GLOBAL_Stream[inp_stream].status) {
|
2015-07-06 12:03:16 +01:00
|
|
|
#if HAVE_FGETPOS
|
2015-08-18 20:37:39 +01:00
|
|
|
fsetpos(GLOBAL_Stream[inp_stream].file, &re->rpos);
|
2015-07-06 12:03:16 +01:00
|
|
|
#else
|
2015-08-18 20:37:39 +01:00
|
|
|
fseek(GLOBAL_Stream[inp_stream].file, re->cpos, 0L);
|
2015-07-06 12:03:16 +01:00
|
|
|
#endif
|
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
return YAP_SCANNING;
|
|
|
|
}
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t parseError(REnv *re, FEnv *fe, int inp_stream) {
|
2015-07-06 12:03:16 +01:00
|
|
|
CACHE_REGS
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->t = 0;
|
2015-09-25 10:57:26 +01:00
|
|
|
if (LOCAL_Error_TYPE == RESOURCE_ERROR_TRAIL ||
|
|
|
|
LOCAL_Error_TYPE == RESOURCE_ERROR_AUXILIARY_STACK ||
|
|
|
|
LOCAL_Error_TYPE == RESOURCE_ERROR_HEAP ||
|
|
|
|
LOCAL_Error_TYPE == RESOURCE_ERROR_STACK) {
|
2015-08-18 20:37:39 +01:00
|
|
|
return YAP_SCANNING_ERROR;
|
|
|
|
}
|
|
|
|
Term ParserErrorStyle = re->sy;
|
2015-07-06 12:03:16 +01:00
|
|
|
if (ParserErrorStyle == TermQuiet) {
|
2015-08-18 20:37:39 +01:00
|
|
|
/* just fail */
|
|
|
|
LOCAL_Error_TYPE = YAP_NO_ERROR;
|
|
|
|
return YAP_PARSING_FINISHED;
|
|
|
|
} else {
|
|
|
|
Term terr = Yap_syntax_error(fe->toklast, inp_stream);
|
|
|
|
if (ParserErrorStyle == TermError) {
|
|
|
|
LOCAL_ErrorMessage = "SYNTAX ERROR";
|
|
|
|
Yap_Error(SYNTAX_ERROR, terr, LOCAL_ErrorMessage);
|
2015-07-06 12:03:16 +01:00
|
|
|
return YAP_PARSING_FINISHED;
|
|
|
|
} else {
|
2015-08-18 20:37:39 +01:00
|
|
|
Yap_PrintWarning(terr);
|
|
|
|
LOCAL_Error_TYPE = YAP_NO_ERROR;
|
|
|
|
if (ParserErrorStyle == TermDec10)
|
|
|
|
return YAP_SCANNING;
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-08-07 22:57:53 +01:00
|
|
|
LOCAL_Error_TYPE = YAP_NO_ERROR;
|
2015-07-06 12:03:16 +01:00
|
|
|
return YAP_PARSING_FINISHED;
|
|
|
|
}
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static parser_state_t parse(REnv *re, FEnv *fe, int inp_stream) {
|
2015-07-06 12:03:16 +01:00
|
|
|
CACHE_REGS
|
2015-08-18 20:37:39 +01:00
|
|
|
TokEntry *tokstart = LOCAL_tokptr;
|
2015-07-23 01:33:30 +01:00
|
|
|
|
2015-07-06 12:03:16 +01:00
|
|
|
fe->t = Yap_Parse(re->prio);
|
2015-07-28 04:22:44 +01:00
|
|
|
fe->toklast = LOCAL_tokptr;
|
|
|
|
LOCAL_tokptr = tokstart;
|
2015-08-18 20:37:39 +01:00
|
|
|
TR = (tr_fr_ptr)tokstart;
|
|
|
|
if (fe->t == 0)
|
2015-07-23 01:33:30 +01:00
|
|
|
return YAP_PARSING_ERROR;
|
2015-08-18 20:37:39 +01:00
|
|
|
if (fe->reading_clause && !complete_clause_processing(fe, tokstart, fe->t))
|
2015-07-23 01:33:30 +01:00
|
|
|
fe->t = 0;
|
2015-08-18 20:37:39 +01:00
|
|
|
else if (!fe->reading_clause && !complete_processing(fe, tokstart))
|
2015-07-23 01:33:30 +01:00
|
|
|
fe->t = 0;
|
2015-06-18 01:20:05 +01:00
|
|
|
#if EMACS
|
|
|
|
first_char = tokstart->TokPos;
|
|
|
|
#endif /* EMACS */
|
2015-07-06 12:03:16 +01:00
|
|
|
return YAP_PARSING_FINISHED;
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief generic routine to read terms from a stream
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* @arg inp_stream: where we read from
|
|
|
|
* @arg: opts, a list with options
|
|
|
|
* @arg: if called from read_term, arity
|
|
|
|
* called from read_clause, -arity
|
|
|
|
*
|
|
|
|
* @return the term or 0 in case of error.
|
|
|
|
*
|
2015-07-28 04:22:44 +01:00
|
|
|
* Implementation uses a state machine: default is init, scan, parse, complete.
|
2015-07-06 12:03:16 +01:00
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
Term Yap_read_term(int inp_stream, Term opts, int nargs) {
|
|
|
|
FEnv fe;
|
2015-07-06 12:03:16 +01:00
|
|
|
REnv re;
|
|
|
|
#if EMACS
|
|
|
|
int emacs_cares = FALSE;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
parser_state_t state = YAP_START_PARSING;
|
2015-08-18 20:37:39 +01:00
|
|
|
while (state != YAP_PARSING_FINISHED) {
|
|
|
|
switch (state) {
|
|
|
|
case YAP_START_PARSING:
|
|
|
|
state = initParser(opts, &fe, &re, inp_stream, nargs);
|
|
|
|
break;
|
|
|
|
case YAP_SCANNING:
|
|
|
|
state = scan(&re, &fe, inp_stream);
|
|
|
|
break;
|
|
|
|
case YAP_SCANNING_ERROR:
|
|
|
|
state = scanError(&re, &fe, inp_stream);
|
|
|
|
break;
|
|
|
|
case YAP_PARSING:
|
|
|
|
state = parse(&re, &fe, inp_stream);
|
|
|
|
break;
|
|
|
|
case YAP_PARSING_ERROR:
|
|
|
|
state = parseError(&re, &fe, inp_stream);
|
|
|
|
break;
|
|
|
|
case YAP_PARSING_FINISHED:
|
|
|
|
return fe.t;
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
return fe.t;
|
|
|
|
}
|
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
static Int
|
2015-08-18 20:37:39 +01:00
|
|
|
read_term2(USES_REGS1) { /* '$read'(+Flag,?Term,?Module,?Vars,-Pos,-Err) */
|
2015-06-18 01:20:05 +01:00
|
|
|
Term rc;
|
2015-07-06 12:03:16 +01:00
|
|
|
yhandle_t h = Yap_InitSlot(ARG1);
|
2015-08-18 20:37:39 +01:00
|
|
|
if ((rc = Yap_read_term(LOCAL_c_input_stream, ARG2, 2)) == 0)
|
2015-06-18 01:20:05 +01:00
|
|
|
return FALSE;
|
2015-08-18 20:37:39 +01:00
|
|
|
return Yap_unify(Yap_GetFromSlot(h), rc);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int read_term(
|
|
|
|
USES_REGS1) { /* '$read2'(+Flag,?Term,?Module,?Vars,-Pos,-Err,+Stream) */
|
2015-06-18 01:20:05 +01:00
|
|
|
int inp_stream;
|
|
|
|
Int out;
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
/* needs to change LOCAL_output_stream for write */
|
2015-07-06 12:03:16 +01:00
|
|
|
yhandle_t h = Yap_InitSlot(ARG2);
|
2015-08-18 20:37:39 +01:00
|
|
|
inp_stream = Yap_CheckStream(ARG1, Input_Stream_f, "read/3");
|
2015-06-18 01:20:05 +01:00
|
|
|
if (inp_stream == -1) {
|
2015-08-18 20:37:39 +01:00
|
|
|
return (FALSE);
|
|
|
|
}
|
|
|
|
out = Yap_read_term(inp_stream, ARG3, 3);
|
2015-07-06 12:03:16 +01:00
|
|
|
UNLOCK(GLOBAL_Stream[inp_stream].streamlock);
|
2015-08-18 20:37:39 +01:00
|
|
|
return out != 0L && Yap_unify(Yap_GetFromSlot(h), out);
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
#define READ_CLAUSE_DEFS() \
|
|
|
|
PAR("comments", filler, READ_CLAUSE_COMMENTS), \
|
|
|
|
PAR("process_comments", boolean, READ_CLAUSE_PROCESS_COMMENTS), \
|
|
|
|
PAR("module", isatom, READ_CLAUSE_MODULE), \
|
|
|
|
PAR("variable_names", filler, READ_CLAUSE_VARIABLE_NAMES), \
|
|
|
|
PAR("term_position", filler, READ_CLAUSE_TERM_POSITION), \
|
|
|
|
PAR("syntax_errors", isatom, READ_CLAUSE_SYNTAX_ERRORS), \
|
|
|
|
PAR(NULL, ok, READ_CLAUSE_END)
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
#define PAR(x, y, z) z
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
typedef enum read_clause_enum_choices {
|
2015-07-06 12:03:16 +01:00
|
|
|
READ_CLAUSE_DEFS()
|
|
|
|
} read_clause_choices_t;
|
|
|
|
|
|
|
|
#undef PAR
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
#define PAR(x, y, z) \
|
|
|
|
{ x, y, z }
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static const param_t read_clause_defs[] = {READ_CLAUSE_DEFS()};
|
2015-07-06 12:03:16 +01:00
|
|
|
#undef PAR
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static xarg *setClauseReadEnv(Term opts, FEnv *fe, struct renv *re,
|
|
|
|
int inp_stream) {
|
2015-07-06 12:03:16 +01:00
|
|
|
CACHE_REGS
|
2015-08-18 20:37:39 +01:00
|
|
|
re->cm = CurrentModule;
|
|
|
|
xarg *args = Yap_ArgListToVector(opts, read_clause_defs, READ_END);
|
2015-07-06 12:03:16 +01:00
|
|
|
if (args == NULL) {
|
2015-08-18 20:37:39 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
re->cm = CurrentModule;
|
|
|
|
re->bq = getBackQuotesFlag();
|
|
|
|
CurrentModule = LOCAL_SourceModule;
|
|
|
|
fe->qq = 0;
|
|
|
|
if (args[READ_CLAUSE_TERM_POSITION].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->tp = args[READ_CLAUSE_TERM_POSITION].tvalue;
|
|
|
|
} else {
|
|
|
|
fe->tp = 0;
|
|
|
|
}
|
|
|
|
if (trueLocalPrologFlag(SINGLE_VAR_WARNINGS_FLAG)) {
|
|
|
|
fe->sp = MkVarTerm();
|
|
|
|
} else {
|
|
|
|
fe->sp = 0;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
if (args[READ_CLAUSE_SYNTAX_ERRORS].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
re->sy = args[READ_CLAUSE_SYNTAX_ERRORS].tvalue;
|
|
|
|
} else {
|
|
|
|
re->sy = TermDec10;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
fe->vp = 0;
|
2015-07-23 01:33:30 +01:00
|
|
|
if (args[READ_CLAUSE_VARIABLE_NAMES].used) {
|
2015-08-18 20:37:39 +01:00
|
|
|
fe->np = args[READ_CLAUSE_VARIABLE_NAMES].tvalue;
|
|
|
|
} else {
|
|
|
|
fe->np = 0;
|
|
|
|
}
|
|
|
|
fe->ce = Yap_CharacterEscapes(CurrentModule);
|
2015-07-06 12:03:16 +01:00
|
|
|
re->seekable = (GLOBAL_Stream[inp_stream].status & Seekable_Stream_f) != 0;
|
|
|
|
if (re->seekable) {
|
2015-08-18 20:37:39 +01:00
|
|
|
if (GLOBAL_Stream[inp_stream].stream_getc == PlUnGetc) {
|
|
|
|
re->had_ungetc = TRUE;
|
|
|
|
re->ungetc_oldc = GLOBAL_Stream[inp_stream].och;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
#if HAVE_FGETPOS
|
2015-08-18 20:37:39 +01:00
|
|
|
fgetpos(GLOBAL_Stream[inp_stream].file, &re->rpos);
|
2015-07-06 12:03:16 +01:00
|
|
|
#else
|
2015-08-18 20:37:39 +01:00
|
|
|
re->cpos = GLOBAL_Stream[inp_stream].charcount;
|
2015-07-06 12:03:16 +01:00
|
|
|
#endif
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
re->prio = LOCAL_default_priority;
|
|
|
|
return args;
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static bool complete_clause_processing(FEnv *fe, TokEntry *tokstart, Term t) {
|
2015-07-06 12:03:16 +01:00
|
|
|
CACHE_REGS
|
2015-09-21 23:05:36 +01:00
|
|
|
Term v1, v2, v3;
|
2015-08-18 20:37:39 +01:00
|
|
|
{
|
2015-09-21 23:05:36 +01:00
|
|
|
fe->old_H = HR;
|
2015-08-18 20:37:39 +01:00
|
|
|
while (TRUE) {
|
|
|
|
fe->old_H = HR;
|
|
|
|
|
|
|
|
if (setjmp(LOCAL_IOBotch) == 0) {
|
|
|
|
v1 = Yap_VarNames(LOCAL_VarTable, TermNil);
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
reset_regs(tokstart, fe);
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
2015-09-21 23:05:36 +01:00
|
|
|
}
|
|
|
|
if (fe->tp) {
|
|
|
|
fe->old_H = HR;
|
|
|
|
while (TRUE) {
|
|
|
|
if (setjmp(LOCAL_IOBotch) == 0) {
|
2015-08-18 20:37:39 +01:00
|
|
|
v2 = MkIntegerTerm(Yap_FirstLineInParse());
|
|
|
|
break;
|
2015-09-21 23:05:36 +01:00
|
|
|
} else {
|
|
|
|
*HR++ = v1;
|
2015-08-18 20:37:39 +01:00
|
|
|
reset_regs(tokstart, fe);
|
2015-09-21 23:05:36 +01:00
|
|
|
v1 = *--HR;
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
|
|
|
if (fe->sp) {
|
2015-09-21 23:05:36 +01:00
|
|
|
fe->old_H = HR;
|
2015-08-18 20:37:39 +01:00
|
|
|
while (TRUE) {
|
|
|
|
fe->old_H = HR;
|
|
|
|
|
|
|
|
if (setjmp(LOCAL_IOBotch) == 0) {
|
|
|
|
v3 = Yap_Singletons(LOCAL_VarTable, TermNil);
|
2015-09-21 23:05:36 +01:00
|
|
|
break;
|
2015-08-18 20:37:39 +01:00
|
|
|
} else {
|
2015-09-21 23:05:36 +01:00
|
|
|
*HR++ = v1;
|
|
|
|
*HR++ = v2;
|
2015-08-18 20:37:39 +01:00
|
|
|
reset_regs(tokstart, fe);
|
2015-09-21 23:05:36 +01:00
|
|
|
v2 = *--HR;
|
|
|
|
v1 = *--HR;
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
|
|
|
Yap_clean_tokenizer(tokstart, LOCAL_VarTable, LOCAL_AnonVarTable);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
if (v3 != TermNil) {
|
2015-09-21 23:05:36 +01:00
|
|
|
Term singls[4];
|
|
|
|
singls[0] = Yap_MkApplTerm(Yap_MkFunctor(AtomSingleton, 1), 1, &v3);
|
|
|
|
singls[1] = MkIntegerTerm(LOCAL_SourceFileLineno);
|
|
|
|
singls[2] = MkAtomTerm(LOCAL_SourceFileName );
|
|
|
|
singls[3] = t;
|
|
|
|
Yap_PrintWarning(
|
|
|
|
Yap_MkApplTerm(Yap_MkFunctor(AtomStyleCheck, 4), 4, singls));
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
|
|
|
if (fe->np && !Yap_unify(v1, fe->np))
|
2015-09-21 23:05:36 +01:00
|
|
|
return 0;
|
2015-08-18 20:37:39 +01:00
|
|
|
if (fe->tp && !Yap_unify(v2, fe->tp))
|
2015-09-21 23:05:36 +01:00
|
|
|
return 0;
|
|
|
|
return fe->t;
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @pred read_clause( +_Stream_, -_Clause_, ?_Opts) is det
|
|
|
|
*
|
2015-07-23 01:33:30 +01:00
|
|
|
u* Same as read_clause/3, but from the standard input stream.
|
2015-07-06 12:03:16 +01:00
|
|
|
*
|
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int read_clause2(USES_REGS1) {
|
2015-06-18 01:20:05 +01:00
|
|
|
Term rc;
|
2015-07-06 12:03:16 +01:00
|
|
|
yhandle_t h = Yap_InitSlot(ARG1);
|
2015-08-18 20:37:39 +01:00
|
|
|
rc = Yap_read_term(LOCAL_c_input_stream, Deref(ARG2), -2);
|
|
|
|
return rc && Yap_unify(Yap_GetFromSlot(h), rc);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @pred read_clause( +_Stream_, -_Clause_, ?_Opts) is det
|
|
|
|
*
|
2015-08-18 20:37:39 +01:00
|
|
|
* This predicate receives a set of options _OPts_ based on read_term/3, but
|
|
|
|
*specific
|
2015-07-06 12:03:16 +01:00
|
|
|
* to readin clauses. The following options are considered:
|
|
|
|
*
|
|
|
|
* + The `comments` option unifies its argument with the comments in the term,
|
|
|
|
* represented as strings
|
|
|
|
* + The `process_comments` option calls a hook, it is current ignored by YAP.
|
|
|
|
* + The `term_position` unifies its argument with a term describing the
|
|
|
|
* position of the term.
|
2015-08-18 20:37:39 +01:00
|
|
|
* + The `syntax_errors` flag controls response to syntactic errors, the
|
|
|
|
*default is `dec10`.
|
2015-07-06 12:03:16 +01:00
|
|
|
*
|
2015-08-18 20:37:39 +01:00
|
|
|
* The next two options are called implicitly:
|
2015-07-06 12:03:16 +01:00
|
|
|
*
|
2015-08-18 20:37:39 +01:00
|
|
|
* + The `module` option is initialised to the current source module, by
|
|
|
|
*default.
|
2015-07-06 12:03:16 +01:00
|
|
|
* + The `tons` option is set from the single var flag
|
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int read_clause(
|
|
|
|
USES_REGS1) { /* '$read2'(+Flag,?Term,?Module,?Vars,-Pos,-Err,+Stream) */
|
2015-06-18 01:20:05 +01:00
|
|
|
int inp_stream;
|
|
|
|
Int out;
|
2015-07-06 12:03:16 +01:00
|
|
|
Term t3 = Deref(ARG3);
|
2015-08-18 20:37:39 +01:00
|
|
|
yhandle_t h = Yap_InitSlot(ARG2);
|
2015-06-18 01:20:05 +01:00
|
|
|
/* needs to change LOCAL_output_stream for write */
|
2015-08-18 20:37:39 +01:00
|
|
|
inp_stream = Yap_CheckStream(ARG1, Input_Stream_f, "read/3");
|
|
|
|
out = Yap_read_term(inp_stream, t3, -3);
|
2015-06-18 01:20:05 +01:00
|
|
|
UNLOCK(GLOBAL_Stream[inp_stream].streamlock);
|
2015-08-18 20:37:39 +01:00
|
|
|
return out && Yap_unify(Yap_GetFromSlot(h), out);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
/**
|
2015-07-06 12:03:16 +01:00
|
|
|
* @pred source_location( - _File_ , _Line_ )
|
|
|
|
*
|
|
|
|
* unify _File_ and _Line_ wuth the position of the last term read, if the term
|
2015-08-18 20:37:39 +01:00
|
|
|
* comes from a stream created by opening a file-system path with open/3 and
|
|
|
|
*friends.>position
|
2015-07-06 12:03:16 +01:00
|
|
|
* It ignores user_input or
|
|
|
|
* sockets.
|
|
|
|
*
|
|
|
|
* @param - _File_
|
|
|
|
* @param - _Line_
|
|
|
|
*
|
|
|
|
* @note SWI-Prolog built-in.
|
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int source_location(USES_REGS1) {
|
|
|
|
return Yap_unify(ARG1, MkAtomTerm(LOCAL_SourceFileName)) &&
|
|
|
|
Yap_unify(ARG2, MkIntegerTerm(LOCAL_SourceFileLineno));
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @pred read(+ _Stream_, - _Term_ ) is iso
|
|
|
|
*
|
|
|
|
* Reads term _T_ from the stream _S_ instead of from the current input
|
|
|
|
* stream.
|
|
|
|
*
|
|
|
|
* @param - _Stream_
|
|
|
|
* @param - _Term_
|
|
|
|
*
|
2015-06-18 01:20:05 +01:00
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int read2(
|
|
|
|
USES_REGS1) { /* '$read2'(+Flag,?Term,?Module,?Vars,-Pos,-Err,+Stream) */
|
2015-06-18 01:20:05 +01:00
|
|
|
int inp_stream;
|
|
|
|
Int out;
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
/* needs to change LOCAL_output_stream for write */
|
2015-08-18 20:37:39 +01:00
|
|
|
inp_stream = Yap_CheckStream(ARG1, Input_Stream_f, "read/3");
|
2015-06-18 01:20:05 +01:00
|
|
|
if (inp_stream == -1) {
|
2015-08-18 20:37:39 +01:00
|
|
|
return (FALSE);
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
out = Yap_read_term(inp_stream, TermNil, 1);
|
2015-07-06 12:03:16 +01:00
|
|
|
UNLOCK(GLOBAL_Stream[inp_stream].streamlock);
|
2015-06-18 01:20:05 +01:00
|
|
|
return out && Yap_unify(ARG2, out);
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
/** @pred read(- _T_) is iso
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
Reads the next term from the current input stream, and unifies it with
|
|
|
|
_T_. The term must be followed by a dot (`.`) and any blank-character
|
|
|
|
as previously defined. The syntax of the term must match the current
|
|
|
|
declarations for operators (see op). If the end-of-stream is reached,
|
|
|
|
_T_ is unified with the atom `end_of_file`. Further reads from of
|
|
|
|
the same stream may cause an error failure (see open/3).
|
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int read1(
|
|
|
|
USES_REGS1) { /* '$read2'(+Flag,?Term,?Module,?Vars,-Pos,-Err,+Stream) */
|
2015-06-18 01:20:05 +01:00
|
|
|
Term out = Yap_read_term(LOCAL_c_input_stream, TermNil, 1);
|
|
|
|
return out && Yap_unify(ARG1, out);
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
/** @pred fileerrors
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
Switches on the file_errors flag so that in certain error conditions
|
|
|
|
Input/Output predicates will produce an appropriated message and abort.
|
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int fileerrors(USES_REGS1) {
|
2015-09-21 23:05:36 +01:00
|
|
|
return setYapFlag(TermFileErrors, TermTrue);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
@pred nofileerrors
|
|
|
|
|
|
|
|
Switches off the `file_errors` flag, so that the predicates see/1,
|
|
|
|
tell/1, open/3 and close/1 just fail, instead of producing
|
|
|
|
an error message and aborting whenever the specified file cannot be
|
|
|
|
opened or closed.
|
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int nofileerrors(
|
|
|
|
USES_REGS1) { /* '$read2'(+Flag,?Term,?Module,?Vars,-Pos,-Err,+Stream) */
|
2015-09-21 23:05:36 +01:00
|
|
|
return setYapFlag(TermFileerrors, TermFalse);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int style_checker(USES_REGS1) {
|
|
|
|
Term t = Deref(ARG1);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
if (IsVarTerm(t)) {
|
2015-08-18 20:37:39 +01:00
|
|
|
Term t = TermNil;
|
|
|
|
if (getYapFlag(MkAtomTerm(AtomSingleVarWarnings)) == TermTrue) {
|
|
|
|
t = MkPairTerm(MkAtomTerm(AtomSingleVarWarnings), t);
|
|
|
|
}
|
|
|
|
if (getYapFlag(MkAtomTerm(AtomDiscontiguousWarnings)) == TermTrue) {
|
|
|
|
t = MkPairTerm(MkAtomTerm(AtomDiscontiguousWarnings), t);
|
|
|
|
}
|
|
|
|
if (getYapFlag(MkAtomTerm(AtomRedefineWarnings)) == TermTrue) {
|
|
|
|
t = MkPairTerm(MkAtomTerm(AtomRedefineWarnings), t);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-08-18 20:37:39 +01:00
|
|
|
} else {
|
|
|
|
while (IsPairTerm(t)) {
|
|
|
|
Term h = HeadOfTerm(t);
|
|
|
|
t = TailOfTerm(t);
|
|
|
|
|
|
|
|
if (IsVarTerm(h)) {
|
|
|
|
Yap_Error(INSTANTIATION_ERROR, t, "style_check/1");
|
|
|
|
return (FALSE);
|
|
|
|
} else if (IsAtomTerm(h)) {
|
|
|
|
Atom at = AtomOfTerm(h);
|
|
|
|
if (at == AtomSingleVarWarnings)
|
|
|
|
setYapFlag(MkAtomTerm(AtomSingleVarWarnings), TermTrue);
|
|
|
|
else if (at == AtomDiscontiguousWarnings)
|
|
|
|
setYapFlag(MkAtomTerm(AtomDiscontiguousWarnings), TermTrue);
|
|
|
|
else if (at == AtomRedefineWarnings)
|
|
|
|
setYapFlag(MkAtomTerm(AtomRedefineWarnings), TermTrue);
|
|
|
|
} else {
|
|
|
|
Atom at = AtomOfTerm(ArgOfTerm(1, h));
|
|
|
|
if (at == AtomSingleVarWarnings)
|
|
|
|
setYapFlag(MkAtomTerm(AtomSingleVarWarnings), TermFalse);
|
|
|
|
else if (at == AtomDiscontiguousWarnings)
|
|
|
|
setYapFlag(MkAtomTerm(AtomDiscontiguousWarnings), TermFalse);
|
|
|
|
else if (at == AtomRedefineWarnings)
|
|
|
|
setYapFlag(MkAtomTerm(AtomRedefineWarnings), TermFalse);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
return TRUE;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
Term Yap_StringToTerm(const char *s, size_t len, encoding_t enc, int prio,
|
|
|
|
Term *bindings) {
|
2015-07-06 12:03:16 +01:00
|
|
|
CACHE_REGS
|
2015-08-18 20:37:39 +01:00
|
|
|
Term bvar = MkVarTerm(), ctl;
|
2015-06-18 01:20:05 +01:00
|
|
|
yhandle_t sl;
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
if (bindings) {
|
2015-08-18 20:37:39 +01:00
|
|
|
ctl = Yap_MkApplTerm(Yap_MkFunctor(AtomVariableNames, 1), 1, &bvar);
|
|
|
|
sl = Yap_InitSlot(bvar);
|
|
|
|
} else {
|
|
|
|
ctl = TermNil;
|
|
|
|
sl = 0;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
Term rval;
|
|
|
|
int stream = Yap_open_buf_read_stream(s, len, enc, MEM_BUF_USER);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
rval = Yap_read_term(stream, ctl, 3);
|
|
|
|
Yap_CloseStream(stream);
|
2015-07-06 12:03:16 +01:00
|
|
|
UNLOCK(GLOBAL_Stream[stream].streamlock);
|
2015-06-19 10:10:02 +01:00
|
|
|
if (rval && bindings) {
|
2015-08-18 20:37:39 +01:00
|
|
|
*bindings = Yap_GetFromSlot(sl);
|
|
|
|
Yap_RecoverSlots(sl, 1 PASS_REGS);
|
|
|
|
}
|
2015-06-18 01:20:05 +01:00
|
|
|
return rval;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
Term Yap_ReadFromAtom(Atom a, Term opts) {
|
2015-06-18 01:20:05 +01:00
|
|
|
Term rval;
|
|
|
|
int sno;
|
2015-08-18 20:37:39 +01:00
|
|
|
if (IsWideAtom(a)) {
|
|
|
|
wchar_t *ws = a->WStrOfAE;
|
|
|
|
size_t len = wcslen(ws);
|
|
|
|
sno = Yap_open_buf_read_stream((char *)ws, len, ENC_ISO_ANSI, MEM_BUF_USER);
|
|
|
|
} else {
|
|
|
|
char *s = a->StrOfAE;
|
|
|
|
size_t len = strlen(s);
|
|
|
|
sno =
|
|
|
|
Yap_open_buf_read_stream((char *)s, len, ENC_ISO_LATIN1, MEM_BUF_USER);
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
rval = Yap_read_term(sno, opts, 3);
|
|
|
|
Yap_CloseStream(sno);
|
|
|
|
return rval;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
static Term readFromBuffer(const char *s, Term opts) {
|
2015-06-18 01:20:05 +01:00
|
|
|
Term rval;
|
|
|
|
int sno;
|
2015-09-21 23:05:36 +01:00
|
|
|
sno = Yap_open_buf_read_stream((char *)s, strlen_utf8((unsigned char *)s), ENC_ISO_UTF8,
|
2015-08-18 20:37:39 +01:00
|
|
|
MEM_BUF_USER);
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-06-18 01:20:05 +01:00
|
|
|
rval = Yap_read_term(sno, opts, 3);
|
|
|
|
Yap_CloseStream(sno);
|
|
|
|
return rval;
|
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
|
|
|
/**
|
2015-09-21 23:05:36 +01:00
|
|
|
* @pred read_term_from_atom( +_Atom_ , - _T_ , + _VarNames_
|
|
|
|
*
|
|
|
|
* read a term _T_ stored in constant _Atom_ and report their names
|
|
|
|
*
|
|
|
|
* @param _Atom_ the source _Atom_
|
|
|
|
* @param _T_ the output term _T_, may be any term
|
|
|
|
* @param _VarNames_ list of _Var_ = _Name_ tuples.
|
|
|
|
*
|
|
|
|
* @notes Originally from SWI-Prolog, in YAP only works with atoms.
|
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int atom_to_term(USES_REGS1) {
|
2015-06-18 01:20:05 +01:00
|
|
|
Term t1 = Deref(ARG1), ctl, rc;
|
|
|
|
Atom at;
|
|
|
|
if (IsVarTerm(t1)) {
|
2015-09-21 23:05:36 +01:00
|
|
|
Yap_Error(INSTANTIATION_ERROR, t1, "atom_to_term/2");
|
2015-08-18 20:37:39 +01:00
|
|
|
return (FALSE);
|
|
|
|
} else if (!IsAtomTerm(t1)) {
|
2015-09-21 23:05:36 +01:00
|
|
|
Yap_Error(TYPE_ERROR_ATOM, t1, "atom_to_term/2");
|
2015-08-18 20:37:39 +01:00
|
|
|
return (FALSE);
|
|
|
|
} else {
|
|
|
|
at = AtomOfTerm(t1);
|
|
|
|
}
|
2015-09-21 23:05:36 +01:00
|
|
|
ctl = TermNil;
|
2015-08-18 20:37:39 +01:00
|
|
|
if ((rc = Yap_ReadFromAtom(at, ctl)) == 0L)
|
2015-09-21 23:05:36 +01:00
|
|
|
return false;
|
2015-08-18 20:37:39 +01:00
|
|
|
return Yap_unify(rc, ARG2);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|
2015-07-06 12:03:16 +01:00
|
|
|
|
2015-09-21 23:05:36 +01:00
|
|
|
/**
|
|
|
|
* @pred string_to_term( ?_Atom_ , ? _T_ )
|
|
|
|
*
|
|
|
|
* read a term _T_ stored in constant _String_, or write the term T as
|
|
|
|
* a constant +Atom
|
|
|
|
*
|
|
|
|
* @param _Atom_ the source _Atom_
|
|
|
|
* @param _T_ the output term _T_, may be any term
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static Int
|
|
|
|
term_to_string(USES_REGS1) {
|
|
|
|
Term t1 = Deref(ARG2), rc = false;
|
|
|
|
const char * s;
|
|
|
|
if (IsVarTerm(t1)) {
|
|
|
|
size_t length;
|
|
|
|
s = Yap_TermToString(ARG2, NULL, 0, &length, 0, Quote_illegal_f|Handle_vars_f);
|
|
|
|
if (!s || !
|
|
|
|
MkStringTerm(s)) {
|
2015-09-25 10:57:26 +01:00
|
|
|
Yap_Error(RESOURCE_ERROR_HEAP,t1,"Could not get memory from the operating system");
|
2015-09-21 23:05:36 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return Yap_unify(ARG2, MkStringTerm(s));
|
|
|
|
} else if (!IsStringTerm(t1)) {
|
|
|
|
Yap_Error(TYPE_ERROR_STRING, t1, "string_to_ter®m/2");
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
s = StringOfTerm(t1);
|
|
|
|
}
|
|
|
|
return (rc = readFromBuffer(s, TermNil)) != 0L && Yap_unify(rc, ARG1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @pred atom_to_term( ?_Atom_ , ? _T_ )
|
|
|
|
*
|
|
|
|
* read a term _T_ stored in constant _Atom_, or write the term T as
|
|
|
|
* a constant +Atom
|
|
|
|
*
|
|
|
|
* @param _Atom_ the source _Atom_
|
|
|
|
* @param _T_ the output term _T_, may be any term
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static Int
|
|
|
|
term_to_atom(USES_REGS1) {
|
|
|
|
Term t1 = Deref(ARG2), ctl, rc = false;
|
|
|
|
Atom at;
|
|
|
|
if (IsVarTerm(t1)) {
|
|
|
|
size_t length;
|
|
|
|
char * s =
|
|
|
|
Yap_TermToString(t1, NULL, 0, &length, 0, Quote_illegal_f|Handle_vars_f);
|
|
|
|
if (!s || !(at = Yap_LookupAtom(s))) {
|
2015-09-25 10:57:26 +01:00
|
|
|
Yap_Error(RESOURCE_ERROR_HEAP,t1,"Could not get memory from the operating system");
|
2015-09-21 23:05:36 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return Yap_unify(ARG2, MkAtomTerm(at));
|
|
|
|
} else if (!IsAtomTerm(t1)) {
|
|
|
|
Yap_Error(TYPE_ERROR_ATOM, t1, "atom_to_term/2");
|
|
|
|
return (FALSE);
|
|
|
|
} else {
|
|
|
|
at = AtomOfTerm(t1);
|
|
|
|
}
|
|
|
|
ctl = TermNil;
|
|
|
|
return Yap_ReadFromAtom(at, ctl) == 0L
|
|
|
|
&& Yap_unify(rc, ARG1);
|
|
|
|
}
|
|
|
|
|
2015-07-06 12:03:16 +01:00
|
|
|
/**
|
|
|
|
* @pred read_term_from_atom( +_Atom_ , - _T_ , + _Options_
|
|
|
|
*
|
|
|
|
* read a term _T_ stored in constant _Atom_ according to _Options_
|
|
|
|
*
|
|
|
|
* @param _Atom_ the source _Atom_
|
|
|
|
* @param _T_ the output term _T_, may be any term
|
|
|
|
* @param _Options_ read_term/3 options.
|
|
|
|
*
|
|
|
|
* @notes Originally from SWI-Prolog, in YAP only works with internalised atoms
|
2015-08-18 20:37:39 +01:00
|
|
|
* Check read_term_from_atomic/3 for the general version. Also, the built-in is
|
|
|
|
*supposed to
|
2015-07-06 12:03:16 +01:00
|
|
|
* use YAP's internal encoding, so please avoid the encoding/1 option.
|
2015-08-18 20:37:39 +01:00
|
|
|
*/
|
|
|
|
static Int read_term_from_atom(USES_REGS1) {
|
2015-07-06 12:03:16 +01:00
|
|
|
Term t1 = Deref(ARG1), rc;
|
|
|
|
Atom at;
|
|
|
|
if (IsVarTerm(t1)) {
|
2015-08-18 20:37:39 +01:00
|
|
|
Yap_Error(INSTANTIATION_ERROR, t1, "style_check/1");
|
|
|
|
return (FALSE);
|
|
|
|
} else if (!IsAtomTerm(t1)) {
|
|
|
|
Yap_Error(TYPE_ERROR_ATOM, t1, "style_check/1");
|
|
|
|
return (FALSE);
|
|
|
|
} else {
|
|
|
|
at = AtomOfTerm(t1);
|
|
|
|
}
|
|
|
|
if ((rc = Yap_ReadFromAtom(at, Deref(ARG3))) == 0L)
|
2015-07-06 12:03:16 +01:00
|
|
|
return false;
|
2015-08-18 20:37:39 +01:00
|
|
|
return Yap_unify(rc, ARG2);
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @pred read_term_from_string( +_String_ , - _T_ , + _Options_
|
|
|
|
*
|
|
|
|
* read a term _T_ stored in constant _String_ according to _Options_
|
|
|
|
*
|
|
|
|
* @param _String_ the source _String_
|
|
|
|
* @param _T_ the output term _T_, may be any term
|
|
|
|
* @param _Options_ read_term/3 options.
|
|
|
|
*
|
|
|
|
* @notes Idea from SWI-Prolog, in YAP only works with strings
|
|
|
|
* Check read_term_from_atomic/3 for the general version.
|
|
|
|
*/
|
2015-08-18 20:37:39 +01:00
|
|
|
static Int read_term_from_string(USES_REGS1) {
|
2015-07-06 12:03:16 +01:00
|
|
|
Term t1 = Deref(ARG1), rc;
|
2015-09-21 23:05:36 +01:00
|
|
|
const unsigned char *s;
|
2015-07-06 12:03:16 +01:00
|
|
|
size_t len;
|
|
|
|
if (IsVarTerm(t1)) {
|
2015-08-18 20:37:39 +01:00
|
|
|
Yap_Error(INSTANTIATION_ERROR, t1, "read_term_from_string/3");
|
|
|
|
return (FALSE);
|
|
|
|
} else if (!IsStringTerm(t1)) {
|
|
|
|
Yap_Error(TYPE_ERROR_STRING, t1, "read_term_from_string/3");
|
|
|
|
return (FALSE);
|
|
|
|
} else {
|
2015-09-21 23:05:36 +01:00
|
|
|
s = UStringOfTerm(t1);
|
|
|
|
len = strlen_utf8(s);
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-09-21 23:05:36 +01:00
|
|
|
char *ss = (char *)s;
|
|
|
|
int sno = Yap_open_buf_read_stream(ss, len, ENC_ISO_UTF8, MEM_BUF_USER);
|
|
|
|
rc = readFromBuffer(ss, Deref(ARG3));
|
2015-07-06 12:03:16 +01:00
|
|
|
Yap_CloseStream(sno);
|
|
|
|
if (!rc)
|
|
|
|
return false;
|
2015-08-18 20:37:39 +01:00
|
|
|
return Yap_unify(rc, ARG2);
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @pred read_term_from_atomic( +_Atomic_ , - _T_ , + _Options_ )
|
|
|
|
*
|
|
|
|
* read a term _T_ stored in text _Atomic_ according to _Options_
|
|
|
|
*
|
2015-08-18 20:37:39 +01:00
|
|
|
* @param _Atomic_ the source may be an atom, string, list of codes, or list of
|
|
|
|
*chars.
|
2015-07-06 12:03:16 +01:00
|
|
|
* @param _T_ the output term _T_, may be any term
|
|
|
|
* @param _Options_ read_term/3 options.
|
|
|
|
*
|
2015-08-18 20:37:39 +01:00
|
|
|
* @notes Idea originally from SWI-Prolog, but in YAP we separate atomic and
|
|
|
|
*atom.
|
2015-07-06 12:03:16 +01:00
|
|
|
* Encoding is fixed in atoms and strings.
|
2015-08-18 20:37:39 +01:00
|
|
|
*/
|
|
|
|
static Int read_term_from_atomic(USES_REGS1) {
|
|
|
|
Term t1 = Deref(ARG1), rc;
|
2015-09-21 23:05:36 +01:00
|
|
|
const unsigned char *s;
|
2015-07-06 12:03:16 +01:00
|
|
|
size_t len;
|
|
|
|
if (IsVarTerm(t1)) {
|
2015-08-18 20:37:39 +01:00
|
|
|
Yap_Error(INSTANTIATION_ERROR, t1, "read_term_from_atomic/3");
|
|
|
|
return (FALSE);
|
|
|
|
} else if (!IsAtomicTerm(t1)) {
|
|
|
|
Yap_Error(TYPE_ERROR_ATOMIC, t1, "read_term_from_atomic/3");
|
|
|
|
return (FALSE);
|
|
|
|
} else {
|
|
|
|
Term t = Yap_AtomicToString(t1 PASS_REGS);
|
2015-09-21 23:05:36 +01:00
|
|
|
s = UStringOfTerm(t);
|
|
|
|
len = strlen_utf8(( unsigned char *)s);
|
2015-08-18 20:37:39 +01:00
|
|
|
}
|
2015-09-21 23:05:36 +01:00
|
|
|
char *ss = (char *)s;
|
|
|
|
int sno = Yap_open_buf_read_stream(ss, len, ENC_ISO_UTF8, MEM_BUF_USER);
|
|
|
|
rc = readFromBuffer(ss, Deref(ARG3));
|
2015-07-06 12:03:16 +01:00
|
|
|
Yap_CloseStream(sno);
|
|
|
|
if (!rc)
|
2015-08-18 20:37:39 +01:00
|
|
|
return false;
|
|
|
|
return Yap_unify(rc, ARG2);
|
2015-07-06 12:03:16 +01:00
|
|
|
}
|
|
|
|
|
2015-08-18 20:37:39 +01:00
|
|
|
void Yap_InitReadTPreds(void) {
|
|
|
|
Yap_InitCPred("read", 1, read1, SyncPredFlag);
|
|
|
|
Yap_InitCPred("read", 2, read2, SyncPredFlag);
|
|
|
|
Yap_InitCPred("read_term", 2, read_term2, SyncPredFlag);
|
2015-09-21 23:05:36 +01:00
|
|
|
Yap_InitCPred("read_term", 3, read_term, 0);
|
2015-08-18 20:37:39 +01:00
|
|
|
Yap_InitCPred("read_clause", 2, read_clause2, SyncPredFlag);
|
2015-09-21 23:05:36 +01:00
|
|
|
Yap_InitCPred("read_clause", 3, read_clause, 0);
|
|
|
|
|
|
|
|
|
|
|
|
Yap_InitCPred("term_to_string", 2, term_to_string, 0);
|
|
|
|
Yap_InitCPred("term_to_atom", 2, term_to_atom, 0);
|
2015-08-18 20:37:39 +01:00
|
|
|
Yap_InitCPred("atom_to_term", 3, atom_to_term, 0);
|
|
|
|
Yap_InitCPred("read_term_from_atom", 3, read_term_from_atom, 0);
|
|
|
|
Yap_InitCPred("read_term_from_atomic", 3, read_term_from_atomic, 0);
|
|
|
|
Yap_InitCPred("read_term_from_string", 3, read_term_from_string, 0);
|
|
|
|
|
|
|
|
Yap_InitCPred("fileerrors", 0, fileerrors, SyncPredFlag);
|
|
|
|
Yap_InitCPred("nofileeleerrors", 0, nofileerrors, SyncPredFlag);
|
|
|
|
Yap_InitCPred("source_location", 2, source_location, SyncPredFlag);
|
|
|
|
Yap_InitCPred("$style_checker", 1, style_checker,
|
|
|
|
SyncPredFlag | HiddenPredFlag);
|
2015-06-18 01:20:05 +01:00
|
|
|
}
|