This repository has been archived on 2023-08-20. You can view files and clone it, but cannot push or open issues or pull requests.
yap-6.3/C/write.c

1142 lines
28 KiB
C
Raw Normal View History

/*************************************************************************
* *
* YAP Prolog *
* *
* Yap Prolog was developed at NCCUP - Universidade do Porto *
* *
* Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997 *
* *
**************************************************************************
* *
* File: write.c *
* Last rev: *
* mods: *
* comments: Writing a Prolog Term *
* *
*************************************************************************/
#ifdef SCCS
static char SccsId[] = "%W% %G%";
#endif
2009-05-22 18:24:30 +01:00
#include <stdlib.h>
#include <math.h>
#include "Yap.h"
#include "Yatom.h"
#include "YapHeap.h"
#include "yapio.h"
2010-11-01 21:28:18 +00:00
#include "clause.h"
#if COROUTINING
#include "attvar.h"
#endif
#if HAVE_STRING_H
#include <string.h>
#endif
#if HAVE_CTYPE_H
#include <ctype.h>
#endif
/* describe the type of the previous term to have been written */
typedef enum {
separator, /* the previous term was a separator like ',', ')', ... */
alphanum, /* the previous term was an atom or number */
symbol /* the previous term was a symbol like +, -, *, .... */
} wtype;
static wtype lastw;
2012-02-13 23:07:31 +00:00
typedef void *wrf;
2009-05-24 21:14:23 +01:00
typedef struct union_slots {
2010-05-10 10:21:56 +01:00
Int old;
Int ptr;
2009-05-24 21:14:23 +01:00
} uslots;
typedef struct union_direct {
2009-05-22 18:24:30 +01:00
Term old;
CELL *ptr;
2009-05-24 21:14:23 +01:00
} udirect;
typedef struct rewind_term {
struct rewind_term *parent;
union {
struct union_slots s;
struct union_direct d;
} u;
2009-05-22 18:24:30 +01:00
} rwts;
typedef struct write_globs {
2012-02-13 23:07:31 +00:00
void *stream;
int Quote_illegal, Ignore_ops, Handle_vars, Use_portray;
2012-02-14 09:10:07 +00:00
int Keep_terms;
2009-05-22 18:24:30 +01:00
int Write_Loops;
2012-02-13 23:07:31 +00:00
int Write_strings;
UInt MaxDepth, MaxArgs;
} wglbs;
STATIC_PROTO(void wrputn, (Int, wrf));
STATIC_PROTO(void wrputf, (Float, wrf));
STATIC_PROTO(void wrputref, (CODEADDR, int, wrf));
STATIC_PROTO(int legalAtom, (unsigned char *));
/*STATIC_PROTO(int LeftOpToProtect, (Atom, int));
STATIC_PROTO(int RightOpToProtect, (Atom, int));*/
STATIC_PROTO(wtype AtomIsSymbols, (unsigned char *));
STATIC_PROTO(void putAtom, (Atom, int, wrf));
2009-05-22 18:24:30 +01:00
STATIC_PROTO(void writeTerm, (Term, int, int, int, struct write_globs *, struct rewind_term *));
2012-02-13 23:07:31 +00:00
#define wrputc(X,WF) Sputcode(X,WF) /* writes a character */
static void
2012-02-13 23:07:31 +00:00
wrputn(Int n, wrf stream) /* writes an integer */
{
CACHE_REGS
char s[256], *s1=s; /* that should be enough for most integers */
if (n < 0) {
if (lastw == symbol)
2012-02-13 23:07:31 +00:00
wrputc(' ', stream);
} else {
if (lastw == alphanum)
2012-02-13 23:07:31 +00:00
wrputc(' ', stream);
}
#if HAVE_SNPRINTF
2010-05-06 15:00:44 +01:00
snprintf(s, 256, Int_FORMAT, n);
#else
2010-05-06 15:00:44 +01:00
sprintf(s, Int_FORMAT, n);
#endif
while (*s1)
2012-02-13 23:07:31 +00:00
wrputc(*s1++, stream);
lastw = alphanum;
}
2012-02-13 23:07:31 +00:00
#define wrputs(s, stream) Sfputs(s, stream)
static void
2012-02-13 23:07:31 +00:00
wrputws(wchar_t *s, wrf stream) /* writes a string */
{
CACHE_REGS
while (*s)
2012-02-13 23:07:31 +00:00
wrputc(*s++, stream);
}
2010-05-27 12:24:15 +01:00
#ifdef USE_GMP
static char *
ensure_space(size_t sz) {
CACHE_REGS
2010-05-27 12:24:15 +01:00
char *s;
s = (char *) Yap_PreAllocCodeSpace();
while (s+sz >= (char *)AuxSp) {
#if USE_SYSTEM_MALLOC
/* may require stack expansion */
if (!Yap_ExpandPreAllocCodeSpace(sz, NULL, TRUE)) {
s = NULL;
break;
}
s = (char *) Yap_PreAllocCodeSpace();
#else
s = NULL;
#endif
}
if (!s) {
s = (char *)TR;
while (s+sz >= LOCAL_TrailTop) {
2010-05-27 12:24:15 +01:00
if (!Yap_growtrail(sz/sizeof(CELL), FALSE)) {
s = NULL;
break;
}
s = (char *)TR;
}
}
if (!s) {
s = (char *)H;
if (s+sz >= (char *)ASP) {
Yap_Error(OUT_OF_STACK_ERROR,TermNil,"not enough space to write bignum: it requires %d bytes", sz);
s = NULL;
}
}
return s;
}
static void
2012-02-13 23:07:31 +00:00
write_mpint(MP_INT *big, wrf stream) {
CACHE_REGS
2010-05-27 12:24:15 +01:00
char *s;
s = ensure_space(3+mpz_sizeinbase(big, 10));
if (mpz_sgn(big) < 0) {
if (lastw == symbol)
2012-02-13 23:07:31 +00:00
wrputc(' ', stream);
2010-05-27 12:24:15 +01:00
} else {
if (lastw == alphanum)
2012-02-13 23:07:31 +00:00
wrputc(' ', stream);
2010-05-27 12:24:15 +01:00
}
if (!s) {
s = mpz_get_str(NULL, 10, big);
if (!s)
return;
2012-02-13 23:07:31 +00:00
wrputs(s,stream);
2010-05-27 12:24:15 +01:00
free(s);
} else {
mpz_get_str(s, 10, big);
2012-02-13 23:07:31 +00:00
wrputs(s,stream);
2010-05-27 12:24:15 +01:00
}
}
#endif
/* writes a bignum */
2010-05-27 12:24:15 +01:00
static void
writebig(Term t, int p, int depth, int rinfixarg, struct write_globs *wglb, struct rewind_term *rwt)
2010-05-27 12:24:15 +01:00
{
2011-07-04 14:14:38 +01:00
CACHE_REGS
2010-05-27 12:24:15 +01:00
CELL *pt = RepAppl(t)+1;
2012-02-14 07:46:37 +00:00
CELL big_tag = pt[0];
#ifdef USE_GMP
2012-02-14 07:46:37 +00:00
if (big_tag == BIG_INT)
2010-05-27 12:24:15 +01:00
{
MP_INT *big = Yap_BigIntOfTerm(t);
2012-02-13 23:07:31 +00:00
write_mpint(big, wglb->stream);
2010-05-27 12:24:15 +01:00
return;
2012-02-14 07:46:37 +00:00
} else if (big_tag == BIG_RATIONAL) {
Term trat = Yap_RatTermToApplTerm(t);
writeTerm(trat, p, depth, rinfixarg, wglb, rwt);
2010-05-27 12:24:15 +01:00
return;
}
#endif
2012-02-14 07:46:37 +00:00
if (big_tag == BLOB_STRING) {
2012-02-13 23:07:31 +00:00
if (wglb->Write_strings)
wrputc('`',wglb->stream);
else
wrputc('"',wglb->stream);
wrputs(Yap_BlobStringOfTerm(t),wglb->stream);
if (wglb->Write_strings)
wrputc('`',wglb->stream);
else
wrputc('"',wglb->stream);
2011-06-21 15:11:07 +01:00
return;
2012-02-14 07:46:37 +00:00
} else if (big_tag == BLOB_WIDE_STRING) {
2011-06-21 15:11:07 +01:00
wchar_t *s = Yap_BlobWideStringOfTerm(t);
2012-02-13 23:07:31 +00:00
if (wglb->Write_strings)
wrputc('`',wglb->stream);
else
wrputc('"', wglb->stream);
2011-06-21 15:11:07 +01:00
while (*s) {
2012-02-13 23:07:31 +00:00
wrputc(*s++, wglb->stream);
2011-06-21 15:11:07 +01:00
}
2012-02-13 23:07:31 +00:00
if (wglb->Write_strings)
wrputc('`',wglb->stream);
else
wrputc('"',wglb->stream);
2011-06-21 15:11:07 +01:00
return;
2012-02-14 07:46:37 +00:00
} else if (big_tag >= USER_BLOB_START && big_tag < USER_BLOB_END) {
Opaque_CallOnWrite f;
CELL blob_info;
blob_info = big_tag - USER_BLOB_START;
if (GLOBAL_OpaqueHandlers &&
(f= GLOBAL_OpaqueHandlers[blob_info].write_handler)) {
(f)(wglb->stream, big_tag, (void *)((MP_INT *)(pt+1)), 0);
}
2011-06-21 15:11:07 +01:00
}
2012-02-13 23:07:31 +00:00
wrputs("0",wglb->stream);
2010-05-27 12:24:15 +01:00
}
static void
2012-02-13 23:07:31 +00:00
wrputf(Float f, wrf stream) /* writes a float */
{
2012-02-13 23:07:31 +00:00
char *format_float(double f, char *buf);
CACHE_REGS
2012-02-13 23:07:31 +00:00
char s[256];
char *buf;
2012-02-13 23:07:31 +00:00
/* use SWI's format_float */
buf = format_float(f, s);
if (!buf) return;
wrputs(buf, stream);
}
static void
2012-02-13 23:07:31 +00:00
wrputref(CODEADDR ref, int Quote_illegal, wrf stream) /* writes a data base reference */
{
char s[256];
2012-02-13 23:07:31 +00:00
putAtom(AtomDBref, Quote_illegal, stream);
2010-01-21 15:05:01 +00:00
#if defined(__linux__) || defined(__APPLE__)
2010-11-01 21:28:18 +00:00
sprintf(s, "(%p," UInt_FORMAT ")", ref, ((LogUpdClause*)ref)->ClRefCount);
#else
2010-11-01 21:28:18 +00:00
sprintf(s, "(0x%p," UInt_FORMAT ")", ref, ((LogUpdClause*)ref)->ClRefCount);
#endif
2012-02-13 23:07:31 +00:00
wrputs(s, stream);
lastw = alphanum;
}
static int
legalAtom(unsigned char *s) /* Is this a legal atom ? */
{
wchar_t ch = *s;
if (ch == '\0')
return FALSE;
if (Yap_chtype[ch] != LC) {
if (ch == '[') {
return (s[1] == ']' && !s[2]);
} else if (ch == '{') {
return (s[1] == '}' && !s[2]);
} else if (Yap_chtype[ch] == SL) {
return (!s[1]);
} else if ((ch == ',' || ch == '.') && !s[1]) {
return FALSE;
} else {
2011-04-22 15:29:41 +01:00
if (ch == '/') {
if (s[1] == '*') return FALSE;
}
while (ch) {
if (Yap_chtype[ch] != SY) {
return FALSE;
}
ch = *++s;
}
}
return TRUE;
} else
while ((ch = *++s) != 0)
if (Yap_chtype[ch] > NU)
return FALSE;
return (TRUE);
}
static wtype
AtomIsSymbols(unsigned char *s) /* Is this atom just formed by symbols ? */
{
int ch;
if (Yap_chtype[(int)s[0]] == SL && s[1] == '\0')
return(separator);
while ((ch = *s++) != '\0') {
if (Yap_chtype[ch] != SY)
return(alphanum);
}
return(symbol);
}
static void
2012-02-13 23:07:31 +00:00
write_quoted(int ch, int quote, wrf stream)
{
CACHE_REGS
if (yap_flags[CHARACTER_ESCAPE_FLAG] == CPROLOG_CHARACTER_ESCAPES) {
2012-02-13 23:07:31 +00:00
wrputc(ch, stream);
if (ch == '\'')
2012-02-13 23:07:31 +00:00
wrputc('\'', stream); /* be careful about quotes */
return;
}
if (!(ch < 0xff && chtype(ch) == BS) && ch != '\'' && ch != '\\') {
2012-02-13 23:07:31 +00:00
wrputc(ch, stream);
} else {
switch (ch) {
case '\\':
case '\'':
2012-02-13 23:07:31 +00:00
wrputc('\\', stream);
wrputc(ch, stream);
break;
case 7:
2012-02-13 23:07:31 +00:00
wrputc('\\', stream);
wrputc('a', stream);
break;
case '\b':
2012-02-13 23:07:31 +00:00
wrputc('\\', stream);
wrputc('b', stream);
break;
case '\t':
2012-02-13 23:07:31 +00:00
wrputc('\\', stream);
wrputc('t', stream);
break;
case ' ':
case 160:
2012-02-13 23:07:31 +00:00
wrputc(' ', stream);
break;
case '\n':
2012-02-13 23:07:31 +00:00
wrputc('\\', stream);
wrputc('n', stream);
break;
case 11:
2012-02-13 23:07:31 +00:00
wrputc('\\', stream);
wrputc('v', stream);
break;
case '\r':
2012-02-13 23:07:31 +00:00
wrputc('\\', stream);
wrputc('r', stream);
break;
case '\f':
2012-02-13 23:07:31 +00:00
wrputc('\\', stream);
wrputc('f', stream);
break;
default:
if ( ch <= 0xff ) {
char esc[8];
if (yap_flags[CHARACTER_ESCAPE_FLAG] == SICSTUS_CHARACTER_ESCAPES) {
sprintf(esc, "\\%03o", ch);
} else {
/* last backslash in ISO mode */
sprintf(esc, "\\%03o\\", ch);
}
2012-02-13 23:07:31 +00:00
wrputs(esc, stream);
}
}
}
}
static void
2012-02-13 23:07:31 +00:00
putAtom(Atom atom, int Quote_illegal, wrf stream) /* writes an atom */
{
CACHE_REGS
unsigned char *s = (unsigned char *)RepAtom(atom)->StrOfAE;
wtype atom_or_symbol = AtomIsSymbols(s);
/* #define CRYPT_FOR_STEVE 1*/
#ifdef CRYPT_FOR_STEVE
if (Yap_GetValue(AtomCryptAtoms) != TermNil && Yap_GetAProp(atom, OpProperty) == NIL) {
char s[16];
sprintf(s,"x%x", (CELL)s);
2012-02-13 23:07:31 +00:00
wrputs(s, stream);
return;
}
#endif
if (IsBlob(atom)) {
2012-02-13 23:07:31 +00:00
wrputref((CODEADDR)RepAtom(atom),1,stream);
return;
}
if (IsWideAtom(atom)) {
wchar_t *ws = (wchar_t *)s;
if (Quote_illegal) {
2012-02-13 23:07:31 +00:00
wrputc('\'', stream);
while (*ws) {
wchar_t ch = *ws++;
2012-02-13 23:07:31 +00:00
write_quoted(ch, '\'', stream);
}
2012-02-13 23:07:31 +00:00
wrputc('\'', stream);
} else {
2012-02-13 23:07:31 +00:00
wrputws(ws, stream);
}
return;
}
if (lastw == atom_or_symbol && atom_or_symbol != separator /* solo */)
2012-02-13 23:07:31 +00:00
wrputc(' ', stream);
lastw = atom_or_symbol;
2011-04-22 15:29:41 +01:00
if (Quote_illegal && !legalAtom(s)) {
2012-02-13 23:07:31 +00:00
wrputc('\'', stream);
while (*s) {
wchar_t ch = *s++;
2012-02-13 23:07:31 +00:00
write_quoted(ch, '\'', stream);
}
2012-02-13 23:07:31 +00:00
wrputc('\'', stream);
} else {
2012-02-13 23:07:31 +00:00
wrputs((char *)s, stream);
}
}
static int
IsStringTerm(Term string) /* checks whether this is a string */
{
if (IsVarTerm(string))
return FALSE;
do {
Term hd;
int ch;
if (!IsPairTerm(string)) return(FALSE);
hd = HeadOfTerm(string);
if (IsVarTerm(hd)) return(FALSE);
if (!IsIntTerm(hd)) return(FALSE);
ch = IntOfTerm(HeadOfTerm(string));
if ((ch < ' ' || ch > MAX_ISO_LATIN1) && ch != '\n' && ch != '\t')
return(FALSE);
string = TailOfTerm(string);
if (IsVarTerm(string)) return(FALSE);
} while (string != TermNil);
return(TRUE);
}
static void
2012-02-13 23:07:31 +00:00
putString(Term string, wrf stream) /* writes a string */
{
CACHE_REGS
2012-02-13 23:07:31 +00:00
wrputc('"', stream);
while (string != TermNil) {
int ch = IntOfTerm(HeadOfTerm(string));
2012-02-13 23:07:31 +00:00
write_quoted(ch, '"', stream);
2008-10-25 09:02:42 +01:00
string = TailOfTerm(string);
}
2012-02-13 23:07:31 +00:00
wrputc('"', stream);
lastw = alphanum;
}
static void
2012-02-13 23:07:31 +00:00
putUnquotedString(Term string, wrf stream) /* writes a string */
{
CACHE_REGS
while (string != TermNil) {
int ch = IntOfTerm(HeadOfTerm(string));
2012-02-13 23:07:31 +00:00
wrputc(ch, stream);
string = TailOfTerm(string);
}
lastw = alphanum;
}
static void
2009-05-22 18:24:30 +01:00
write_var(CELL *t, struct write_globs *wglb, struct rewind_term *rwt)
{
CACHE_REGS
if (lastw == alphanum) {
2012-02-13 23:07:31 +00:00
wrputc(' ', wglb->stream);
}
2012-02-13 23:07:31 +00:00
wrputc('_', wglb->stream);
/* make sure we don't get no creepy spaces where they shouldn't be */
lastw = separator;
2010-03-08 09:23:58 +00:00
if (IsAttVar(t)) {
Int vcount = (t-H0);
#if COROUTINING
#if DEBUG
if (Yap_Portray_delays) {
exts ext = ExtFromCell(t);
Yap_Portray_delays = FALSE;
if (ext == attvars_ext) {
2010-03-08 09:23:58 +00:00
attvar_record *attv = RepAttVar(t);
2010-05-10 10:21:56 +01:00
Int sl = 0;
Term l = attv->Atts;
2012-02-13 23:07:31 +00:00
wrputs("$AT(",wglb->stream);
2009-05-22 18:24:30 +01:00
write_var(t, wglb, rwt);
2012-02-13 23:07:31 +00:00
wrputc(',', wglb->stream);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
sl = Yap_InitSlot((CELL)attv PASS_REGS);
}
2009-05-22 18:24:30 +01:00
writeTerm((Term)&(attv->Value), 999, 1, FALSE, wglb, rwt);
2012-02-13 23:07:31 +00:00
wrputc(',', wglb->stream);
2009-05-22 18:24:30 +01:00
writeTerm(l, 999, 1, FALSE, wglb, rwt);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
attv = (attvar_record *)Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
}
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
}
Yap_Portray_delays = TRUE;
return;
}
#endif
2012-02-13 23:07:31 +00:00
wrputc('D', wglb->stream);
wrputn(vcount,wglb->stream);
#endif
} else {
2012-02-13 23:07:31 +00:00
wrputn(((Int) (t- H0)),wglb->stream);
}
}
2009-05-22 18:24:30 +01:00
static Term
2009-05-24 21:14:23 +01:00
from_pointer(CELL *ptr, struct rewind_term *rwt, struct write_globs *wglb)
2009-05-22 18:24:30 +01:00
{
CACHE_REGS
2009-05-22 18:24:30 +01:00
Term t;
2009-05-24 21:14:23 +01:00
while (IsVarTerm(*ptr) && !IsUnboundVar(ptr))
2009-05-22 18:24:30 +01:00
ptr = (CELL *)*ptr;
t = *ptr;
2010-11-01 20:10:32 +00:00
if (!IsVarTerm(t) && !IsAtomOrIntTerm(t)) {
struct rewind_term *x = rwt->parent;
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
rwt->u.s.old = Yap_InitSlot(t PASS_REGS);
rwt->u.s.ptr = Yap_InitSlot((CELL)ptr PASS_REGS);
2010-11-01 20:10:32 +00:00
while (x) {
if (Yap_GetFromSlot(x->u.s.old PASS_REGS) == t)
2010-11-01 20:10:32 +00:00
return TermFoundVar;
x = x->parent;
}
2009-05-24 21:14:23 +01:00
} else {
rwt->u.d.old = t;
rwt->u.d.ptr = ptr;
2010-11-01 20:10:32 +00:00
while (x) {
if (x->u.d.old == t)
return TermFoundVar;
x = x->parent;
}
}
} else {
rwt->u.s.ptr = 0;
}
return t;
}
static Term
check_infinite_loop(Term t, struct rewind_term *x, struct write_globs *wglb)
{
CACHE_REGS
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
2010-11-01 20:10:32 +00:00
while (x) {
if (Yap_GetFromSlot(x->u.s.old PASS_REGS) == t)
2010-11-01 20:10:32 +00:00
return TermFoundVar;
x = x->parent;
}
} else {
while (x) {
if (x->u.d.old == t)
return TermFoundVar;
x = x->parent;
2009-05-24 21:14:23 +01:00
}
}
2009-05-22 18:24:30 +01:00
return t;
}
2009-05-24 21:14:23 +01:00
static void
restore_from_write(struct rewind_term *rwt, struct write_globs *wglb)
{
CACHE_REGS
2009-05-24 21:14:23 +01:00
Term t;
if (rwt->u.s.ptr) {
CELL *ptr;
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
ptr = (CELL *)Yap_GetPtrFromSlot(rwt->u.s.ptr PASS_REGS);
t = Yap_GetPtrFromSlot(rwt->u.s.old PASS_REGS);
Yap_RecoverSlots(2 PASS_REGS);
2009-05-24 21:14:23 +01:00
} else {
ptr = rwt->u.d.ptr;
t = rwt->u.d.old;
}
}
rwt->u.s.ptr = 0;
}
static void
write_list(Term t, int direction, int depth, struct write_globs *wglb, struct rewind_term *rwt)
{
CACHE_REGS
2009-05-24 21:14:23 +01:00
Term ti;
struct rewind_term nrwt;
nrwt.parent = rwt;
nrwt.u.s.ptr = 0;
while (1) {
2010-05-10 10:21:56 +01:00
Int sl= 0;
2009-05-24 21:14:23 +01:00
int ndirection;
int do_jump;
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
2009-05-24 21:14:23 +01:00
/* garbage collection may be called */
sl = Yap_InitSlot(t PASS_REGS);
2009-05-24 21:14:23 +01:00
}
writeTerm(from_pointer(RepPair(t), &nrwt, wglb), 999, depth+1, FALSE, wglb, &nrwt);
2009-05-24 21:14:23 +01:00
restore_from_write(&nrwt, wglb);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
2009-05-24 21:14:23 +01:00
}
ti = TailOfTerm(t);
if (IsVarTerm(ti))
break;
2010-11-01 20:10:32 +00:00
if (!IsPairTerm(ti) ||
!IsPairTerm((ti = check_infinite_loop(ti, rwt, wglb))))
2009-05-24 21:14:23 +01:00
break;
ndirection = RepPair(ti)-RepPair(t);
/* make sure we're not trapped in loops */
if (ndirection > 0) {
2010-11-01 20:10:32 +00:00
do_jump = (direction <= 0);
2009-05-24 21:14:23 +01:00
} else if (ndirection == 0) {
2012-02-13 23:07:31 +00:00
wrputc(',', wglb->stream);
putAtom(AtomFoundVar, wglb->Quote_illegal, wglb->stream);
2009-05-24 21:14:23 +01:00
lastw = separator;
return;
} else {
2010-11-01 20:10:32 +00:00
do_jump = (direction >= 0);
2009-05-24 21:14:23 +01:00
}
2009-06-02 04:54:56 +01:00
if (wglb->MaxDepth != 0 && depth > wglb->MaxDepth) {
2012-02-13 23:07:31 +00:00
wrputc('|', wglb->stream);
putAtom(Atom3Dots, wglb->Quote_illegal, wglb->stream);
2009-06-02 04:54:56 +01:00
return;
}
2009-05-24 21:14:23 +01:00
lastw = separator;
direction = ndirection;
depth++;
2009-05-24 21:14:23 +01:00
if (do_jump)
break;
2012-02-13 23:07:31 +00:00
wrputc(',', wglb->stream);
2009-05-24 21:14:23 +01:00
t = ti;
}
if (IsPairTerm(ti)) {
2010-11-01 20:10:32 +00:00
Term nt = from_pointer(RepPair(t)+1, &nrwt, wglb);
/* we found an infinite loop */
if (IsAtomTerm(nt)) {
2012-02-13 23:07:31 +00:00
wrputc('|', wglb->stream);
2010-11-01 20:10:32 +00:00
writeTerm(nt, 999, depth, FALSE, wglb, rwt);
} else {
/* keep going on the list */
2012-02-13 23:07:31 +00:00
wrputc(',', wglb->stream);
2010-11-01 20:10:32 +00:00
write_list(nt, direction, depth, wglb, &nrwt);
}
2009-05-24 21:14:23 +01:00
restore_from_write(&nrwt, wglb);
} else if (ti != MkAtomTerm(AtomNil)) {
2012-02-13 23:07:31 +00:00
wrputc('|', wglb->stream);
2009-05-24 21:14:23 +01:00
lastw = separator;
writeTerm(from_pointer(RepPair(t)+1, &nrwt, wglb), 999, depth, FALSE, wglb, &nrwt);
2009-05-24 21:14:23 +01:00
restore_from_write(&nrwt, wglb);
}
}
static void
2009-05-22 18:24:30 +01:00
writeTerm(Term t, int p, int depth, int rinfixarg, struct write_globs *wglb, struct rewind_term *rwt)
2009-05-24 21:14:23 +01:00
/* term to write */
/* context priority */
{
CACHE_REGS
2009-05-22 18:24:30 +01:00
struct rewind_term nrwt;
nrwt.parent = rwt;
2009-05-24 21:14:23 +01:00
nrwt.u.s.ptr = 0;
2009-05-22 18:24:30 +01:00
if (wglb->MaxDepth != 0 && depth > wglb->MaxDepth) {
2012-02-13 23:07:31 +00:00
putAtom(Atom3Dots, wglb->Quote_illegal, wglb->stream);
return;
}
2010-07-27 23:29:55 +01:00
if (EX)
return;
t = Deref(t);
if (IsVarTerm(t)) {
2009-05-22 18:24:30 +01:00
write_var((CELL *)t, wglb, &nrwt);
} else if (IsIntTerm(t)) {
2012-02-13 23:07:31 +00:00
wrputn((Int) IntOfTerm(t),wglb->stream);
} else if (IsAtomTerm(t)) {
2012-02-13 23:07:31 +00:00
putAtom(AtomOfTerm(t), wglb->Quote_illegal, wglb->stream);
} else if (IsPairTerm(t)) {
2011-04-22 15:29:41 +01:00
if (wglb->Ignore_ops) {
Int sl = 0;
2012-02-13 23:07:31 +00:00
wrputs("'.'(",wglb->stream);
2011-04-22 15:29:41 +01:00
lastw = separator;
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
2011-04-22 15:29:41 +01:00
/* garbage collection may be called */
sl = Yap_InitSlot(t PASS_REGS);
2011-04-22 15:29:41 +01:00
}
writeTerm(HeadOfTerm(t), 999, depth + 1, FALSE, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
2011-04-22 15:29:41 +01:00
/* garbage collection may be called */
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
2011-04-22 15:29:41 +01:00
}
2012-02-13 23:07:31 +00:00
wrputs(",",wglb->stream);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
2011-04-22 15:29:41 +01:00
/* garbage collection may be called */
sl = Yap_InitSlot(t PASS_REGS);
2011-04-22 15:29:41 +01:00
}
writeTerm(TailOfTerm(t), 999, depth + 1, FALSE, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
2011-04-22 15:29:41 +01:00
/* garbage collection may be called */
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
2011-04-22 15:29:41 +01:00
}
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
2011-04-22 15:29:41 +01:00
lastw = separator;
return;
}
if (wglb->Use_portray) {
Term targs[1];
2010-07-27 23:29:55 +01:00
struct DB_TERM *old_EX = NULL;
2010-05-10 10:21:56 +01:00
Int sl = 0;
targs[0] = t;
Yap_PutValue(AtomPortray, MkAtomTerm(AtomNil));
2010-07-27 23:29:55 +01:00
if (EX) old_EX = EX;
sl = Yap_InitSlot(t PASS_REGS);
Yap_execute_goal(Yap_MkApplTerm(FunctorPortray, 1, targs), 0, 1);
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
2010-07-27 23:29:55 +01:00
if (old_EX != NULL) EX = old_EX;
if (Yap_GetValue(AtomPortray) == MkAtomTerm(AtomTrue))
return;
}
if (yap_flags[WRITE_QUOTED_STRING_FLAG] && IsStringTerm(t)) {
2012-02-13 23:07:31 +00:00
putString(t, wglb->stream);
} else {
2012-02-13 23:07:31 +00:00
wrputc('[', wglb->stream);
lastw = separator;
2010-11-01 20:10:32 +00:00
/* we assume t was already saved in the stack */
write_list(t, 0, depth, wglb, rwt);
2012-02-13 23:07:31 +00:00
wrputc(']', wglb->stream);
lastw = separator;
}
} else { /* compound term */
Functor functor = FunctorOfTerm(t);
int Arity;
Atom atom;
int op, lp, rp;
if (IsExtensionFunctor(functor)) {
switch((CELL)functor) {
case (CELL)FunctorDouble:
2012-02-13 23:07:31 +00:00
wrputf(FloatOfTerm(t),wglb->stream);
return;
2010-03-08 09:23:58 +00:00
case (CELL)FunctorAttVar:
write_var(RepAppl(t)+1, wglb, &nrwt);
return;
case (CELL)FunctorDBRef:
2012-02-13 23:07:31 +00:00
wrputref(RefOfTerm(t), wglb->Quote_illegal, wglb->stream);
return;
case (CELL)FunctorLongInt:
2012-02-13 23:07:31 +00:00
wrputn(LongIntOfTerm(t),wglb->stream);
return;
/* case (CELL)FunctorBigInt: */
default:
writebig(t, p, depth, rinfixarg, wglb, rwt);
return;
}
}
Arity = ArityOfFunctor(functor);
atom = NameOfFunctor(functor);
#ifdef SFUNC
if (Arity == SFArity) {
int argno = 1;
CELL *p = ArgsOfSFTerm(t);
2012-02-13 23:07:31 +00:00
putAtom(atom, wglb->Quote_illegal, wglb->stream);
wrputc('(', wglb->stream);
lastw = separator;
while (*p) {
2010-05-10 10:21:56 +01:00
Int sl = 0;
while (argno < *p) {
2012-02-13 23:07:31 +00:00
wrputc('_', wglb->stream), wrputc(',', wglb->stream);
++argno;
}
*p++;
lastw = separator;
/* cannot use the term directly with the SBA */
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
sl = Yap_InitSlot((CELL)p);
}
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(p++, &nrwt, wglb), 999, depth + 1, FALSE, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
p = (CELL *)Yap_GetFromSlot(sl);
Yap_RecoverSlots(1);
}
if (*p)
2012-02-13 23:07:31 +00:00
wrputc(',', wglb->stream);
argno++;
}
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
return;
}
#endif
if (wglb->Use_portray) {
Term targs[1];
2010-07-27 23:29:55 +01:00
struct DB_TERM *old_EX = NULL;
2010-05-10 10:21:56 +01:00
Int sl = 0;
targs[0] = t;
Yap_PutValue(AtomPortray, MkAtomTerm(AtomNil));
2010-07-27 23:29:55 +01:00
if (EX) old_EX = EX;
sl = Yap_InitSlot(t PASS_REGS);
Yap_execute_goal(Yap_MkApplTerm(FunctorPortray, 1, targs),0, 1);
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
2010-07-27 23:29:55 +01:00
if (old_EX) EX = old_EX;
if (Yap_GetValue(AtomPortray) == MkAtomTerm(AtomTrue) || EX)
return;
}
if (!wglb->Ignore_ops &&
2009-11-20 00:33:14 +00:00
Arity == 1 && Yap_IsPrefixOp(atom, &op, &rp)
#ifdef DO_NOT_WRITE_PLUS_AND_MINUS_AS_PREFIX
&&
/* never write '+' and '-' as infix
operators */
( (RepAtom(atom)->StrOfAE[0] != '+' &&
RepAtom(atom)->StrOfAE[0] != '-') ||
RepAtom(atom)->StrOfAE[1] )
#endif /* DO_NOT_WRITE_PLUS_AND_MINUS_AS_PREFIX */
) {
Term tright = ArgOfTerm(1, t);
int bracket_right =
!IsVarTerm(tright) && IsAtomTerm(tright) &&
Yap_IsOp(AtomOfTerm(tright));
if (op > p) {
/* avoid stuff such as \+ (a,b) being written as \+(a,b) */
if (lastw != separator && !rinfixarg)
2012-02-13 23:07:31 +00:00
wrputc(' ', wglb->stream);
wrputc('(', wglb->stream);
lastw = separator;
}
2012-02-13 23:07:31 +00:00
putAtom(atom, wglb->Quote_illegal, wglb->stream);
if (bracket_right) {
2012-02-13 23:07:31 +00:00
wrputc('(', wglb->stream);
lastw = separator;
}
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(RepAppl(t)+1, &nrwt, wglb), rp, depth + 1, FALSE, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
if (bracket_right) {
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
}
if (op > p) {
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
}
} else if (!wglb->Ignore_ops &&
2009-11-20 00:33:14 +00:00
Arity == 1 &&
Yap_IsPosfixOp(atom, &op, &lp)) {
Term tleft = ArgOfTerm(1, t);
2010-05-10 10:21:56 +01:00
Int sl = 0;
int bracket_left =
!IsVarTerm(tleft) && IsAtomTerm(tleft) &&
Yap_IsOp(AtomOfTerm(tleft));
if (op > p) {
/* avoid stuff such as \+ (a,b) being written as \+(a,b) */
if (lastw != separator && !rinfixarg)
2012-02-13 23:07:31 +00:00
wrputc(' ', wglb->stream);
wrputc('(', wglb->stream);
lastw = separator;
}
if (bracket_left) {
2012-02-13 23:07:31 +00:00
wrputc('(', wglb->stream);
lastw = separator;
}
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
sl = Yap_InitSlot(t PASS_REGS);
}
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(RepAppl(t)+1, &nrwt, wglb), lp, depth + 1, rinfixarg, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
}
if (bracket_left) {
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
}
2012-02-13 23:07:31 +00:00
putAtom(atom, wglb->Quote_illegal, wglb->stream);
if (op > p) {
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
}
} else if (!wglb->Ignore_ops &&
2009-11-20 00:33:14 +00:00
Arity == 2 && Yap_IsInfixOp(atom, &op, &lp,
&rp) ) {
Term tleft = ArgOfTerm(1, t);
Term tright = ArgOfTerm(2, t);
2010-05-10 10:21:56 +01:00
Int sl = 0;
int bracket_left =
!IsVarTerm(tleft) && IsAtomTerm(tleft) &&
Yap_IsOp(AtomOfTerm(tleft));
int bracket_right =
!IsVarTerm(tright) && IsAtomTerm(tright) &&
Yap_IsOp(AtomOfTerm(tright));
if (op > p) {
/* avoid stuff such as \+ (a,b) being written as \+(a,b) */
if (lastw != separator && !rinfixarg)
2012-02-13 23:07:31 +00:00
wrputc(' ', wglb->stream);
wrputc('(', wglb->stream);
lastw = separator;
}
if (bracket_left) {
2012-02-13 23:07:31 +00:00
wrputc('(', wglb->stream);
lastw = separator;
}
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
sl = Yap_InitSlot(t PASS_REGS);
}
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(RepAppl(t)+1, &nrwt, wglb), lp, depth + 1, rinfixarg, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
}
if (bracket_left) {
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
}
/* avoid quoting commas */
if (strcmp(RepAtom(atom)->StrOfAE,","))
2012-02-13 23:07:31 +00:00
putAtom(atom, wglb->Quote_illegal, wglb->stream);
else {
2012-02-13 23:07:31 +00:00
wrputc(',', wglb->stream);
lastw = separator;
}
if (bracket_right) {
2012-02-13 23:07:31 +00:00
wrputc('(', wglb->stream);
lastw = separator;
}
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(RepAppl(t)+2, &nrwt, wglb), rp, depth + 1, TRUE, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
if (bracket_right) {
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
}
if (op > p) {
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
}
} else if (wglb->Handle_vars && functor == FunctorVar) {
Term ti = ArgOfTerm(1, t);
if (lastw == alphanum) {
2012-02-13 23:07:31 +00:00
wrputc(' ', wglb->stream);
}
2012-02-14 07:46:37 +00:00
if (!IsVarTerm(ti) && (IsIntTerm(ti) || IsStringTerm(ti) || IsAtomTerm(ti))) {
if (IsIntTerm(ti)) {
Int k = IntOfTerm(ti);
if (k == -1) {
2012-02-13 23:07:31 +00:00
wrputc('_', wglb->stream);
lastw = alphanum;
return;
} else {
2012-02-13 23:07:31 +00:00
wrputc((k % 26) + 'A', wglb->stream);
if (k >= 26) {
/* make sure we don't get confused about our context */
lastw = separator;
2012-02-13 23:07:31 +00:00
wrputn( k / 26 ,wglb->stream);
} else
lastw = alphanum;
}
2012-02-14 06:54:32 +00:00
} else if (IsAtomTerm(ti)) {
putAtom(AtomOfTerm(ti), FALSE, wglb->stream);
} else {
2012-02-13 23:07:31 +00:00
putUnquotedString(ti, wglb->stream);
}
} else {
2010-05-10 10:21:56 +01:00
Int sl = 0;
2012-02-13 23:07:31 +00:00
wrputs("'$VAR'(",wglb->stream);
lastw = separator;
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
sl = Yap_InitSlot(t PASS_REGS);
}
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(RepAppl(t)+1, &nrwt, wglb), 999, depth + 1, FALSE, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
}
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
}
} else if (!wglb->Ignore_ops && functor == FunctorBraces) {
2012-02-13 23:07:31 +00:00
wrputc('{', wglb->stream);
lastw = separator;
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(RepAppl(t)+1, &nrwt, wglb), 1200, depth + 1, FALSE, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
2012-02-13 23:07:31 +00:00
wrputc('}', wglb->stream);
lastw = separator;
} else if (atom == AtomArray) {
2010-05-10 10:21:56 +01:00
Int sl = 0;
2012-02-13 23:07:31 +00:00
wrputc('{', wglb->stream);
lastw = separator;
for (op = 1; op <= Arity; ++op) {
if (op == wglb->MaxArgs) {
2012-02-14 07:46:37 +00:00
wrputs("...", wglb->stream);
break;
}
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
sl = Yap_InitSlot(t PASS_REGS);
}
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(RepAppl(t)+op, &nrwt, wglb), 999, depth + 1, FALSE, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
}
if (op != Arity) {
2012-02-13 23:07:31 +00:00
wrputc(',', wglb->stream);
lastw = separator;
}
}
2012-02-13 23:07:31 +00:00
wrputc('}', wglb->stream);
lastw = separator;
} else {
2012-02-13 23:07:31 +00:00
putAtom(atom, wglb->Quote_illegal, wglb->stream);
lastw = separator;
2012-02-13 23:07:31 +00:00
wrputc('(', wglb->stream);
for (op = 1; op <= Arity; ++op) {
2010-05-10 10:21:56 +01:00
Int sl = 0;
if (op == wglb->MaxArgs) {
2012-02-13 23:07:31 +00:00
wrputc('.', wglb->stream);
wrputc('.', wglb->stream);
wrputc('.', wglb->stream);
break;
}
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
sl = Yap_InitSlot(t PASS_REGS);
}
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(RepAppl(t)+op, &nrwt, wglb), 999, depth + 1, FALSE, wglb, &nrwt);
restore_from_write(&nrwt, wglb);
2012-02-14 09:10:07 +00:00
if (wglb->Keep_terms) {
/* garbage collection may be called */
t = Yap_GetFromSlot(sl PASS_REGS);
Yap_RecoverSlots(1 PASS_REGS);
}
if (op != Arity) {
2012-02-13 23:07:31 +00:00
wrputc(',', wglb->stream);
lastw = separator;
}
}
2012-02-13 23:07:31 +00:00
wrputc(')', wglb->stream);
lastw = separator;
}
}
}
void
2012-02-14 07:46:37 +00:00
Yap_plwrite(Term t, void *mywrite, int max_depth, int flags, int priority)
/* term to be written */
/* consumer */
/* write options */
{
struct write_globs wglb;
2009-05-22 18:24:30 +01:00
struct rewind_term rwt;
2012-02-13 23:07:31 +00:00
if (!mywrite)
wglb.stream = Serror;
else
wglb.stream = mywrite;
lastw = separator;
wglb.Quote_illegal = flags & Quote_illegal_f;
wglb.Handle_vars = flags & Handle_vars_f;
wglb.Use_portray = flags & Use_portray_f;
2012-02-14 07:46:37 +00:00
wglb.MaxDepth = max_depth;
wglb.MaxArgs = max_depth;
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
2012-02-14 09:10:07 +00:00
wglb.Keep_terms = (flags & (Use_portray_f|To_heap_f));
2010-11-01 20:10:32 +00:00
/* initialise wglb */
rwt.parent = NULL;
wglb.Ignore_ops = flags & Ignore_ops_f;
2012-02-13 23:07:31 +00:00
wglb.Write_strings = flags & BackQuote_String_f;
/* protect slots for portray */
2009-05-24 21:14:23 +01:00
writeTerm(from_pointer(&t, &rwt, &wglb), priority, 1, FALSE, &wglb, &rwt);
restore_from_write(&rwt, &wglb);
}