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/os/writeterm.c

763 lines
22 KiB
C
Raw Normal View History

2015-06-18 01:40:15 +01:00
/*************************************************************************
2017-09-23 02:17:55 +01:00
* *
* YAP Prolog *
* *
* 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:40:15 +01:00
#ifdef SCCS
static char SccsId[] = "%W% %G%";
#endif
/*
* This file includes the definition of a miscellania of standard predicates
2016-05-12 11:44:32 +01:00
* for yap refering to: Files and GLOBAL_Streams, Simple Input/Output,
2015-06-18 01:40:15 +01:00
*
*/
#include "Yap.h"
2017-09-23 02:17:55 +01:00
#include "YapEval.h"
2015-06-18 01:40:15 +01:00
#include "YapHeap.h"
#include "YapText.h"
2016-05-12 11:44:32 +01:00
#include "Yatom.h"
#include "yapio.h"
2015-06-18 01:40:15 +01:00
#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
2016-05-12 11:44:32 +01:00
#if HAVE_SYS_SELECT_H && !_MSC_VER && !defined(__MINGW32__)
2015-06-18 01:40:15 +01:00
#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
2016-05-12 11:44:32 +01:00
#define strncat(X, Y, Z) strcat(X, Y)
2015-06-18 01:40:15 +01:00
#endif
#if !HAVE_STRNCPY
2016-05-12 11:44:32 +01:00
#define strncpy(X, Y, Z) strcpy(X, Y)
2015-06-18 01:40:15 +01:00
#endif
2016-05-12 11:44:32 +01:00
#if _MSC_VER || defined(__MINGW32__)
2015-06-18 01:40:15 +01:00
#if HAVE_SOCKET
#include <winsock2.h>
#endif
#include <windows.h>
#ifndef S_ISDIR
2016-05-12 11:44:32 +01:00
#define S_ISDIR(x) (((x)&_S_IFDIR) == _S_IFDIR)
2015-06-18 01:40:15 +01:00
#endif
#endif
#include "iopreds.h"
2016-07-31 16:17:10 +01:00
static Term readFromBuffer(const char *s, Term opts) {
Term rval;
int sno;
encoding_t enc = ENC_ISO_UTF8;
2018-07-21 12:08:09 +01:00
sno = Yap_open_buf_read_stream(
(char *)s, strlen_utf8((unsigned char *)s), &enc, MEM_BUF_USER,
Yap_LookupAtom(Yap_StrPrefix((char *)s, 16)), TermNone);
2016-07-31 16:17:10 +01:00
rval = Yap_read_term(sno, opts, 3);
Yap_CloseStream(sno);
return rval;
}
2016-10-20 04:44:59 +01:00
2016-05-12 11:44:32 +01:00
#if _MSC_VER || defined(__MINGW32__)
2015-06-18 01:40:15 +01:00
#define SYSTEM_STAT _stat
#else
#define SYSTEM_STAT stat
#endif
#undef PAR
2016-05-12 11:44:32 +01:00
#define WRITE_DEFS() \
PAR("module", isatom, WRITE_MODULE) \
, PAR("attributes", isatom, WRITE_ATTRIBUTES), \
PAR("cycles", booleanFlag, WRITE_CYCLES), \
PAR("quoted", booleanFlag, WRITE_QUOTED), \
PAR("ignore_ops", booleanFlag, WRITE_IGNORE_OPS), \
PAR("max_depth", nat, WRITE_MAX_DEPTH), \
PAR("numbervars", booleanFlag, WRITE_NUMBERVARS), \
PAR("portrayed", booleanFlag, WRITE_PORTRAYED), \
PAR("portray", booleanFlag, WRITE_PORTRAY), \
PAR("priority", nat, WRITE_PRIORITY), \
PAR("character_escapes", booleanFlag, WRITE_CHARACTER_ESCAPES), \
PAR("backquotes", booleanFlag, WRITE_BACKQUOTES), \
PAR("brace_terms", booleanFlag, WRITE_BRACE_TERMS), \
PAR("fullstop", booleanFlag, WRITE_FULLSTOP), \
PAR("nl", booleanFlag, WRITE_NL), \
PAR("variable_names", ok, WRITE_VARIABLE_NAMES), \
PAR(NULL, ok, WRITE_END)
#define PAR(x, y, z) z
typedef enum open_enum_choices { WRITE_DEFS() } open_choices_t;
2015-06-18 01:40:15 +01:00
#undef PAR
2016-05-12 11:44:32 +01:00
#define PAR(x, y, z) \
{ x, y, z }
2015-06-18 01:40:15 +01:00
2016-05-12 11:44:32 +01:00
static const param_t write_defs[] = {WRITE_DEFS()};
#undef PAR
2015-06-18 01:40:15 +01:00
#ifdef BEAM
2016-05-12 11:44:32 +01:00
int beam_write(USES_REGS1) {
2015-06-18 01:40:15 +01:00
Yap_StartSlots();
2016-05-12 11:44:32 +01:00
Yap_plwrite(ARG1, GLOBAL_Stream + LOCAL_c_output_stream, 0, 0,
GLOBAL_MaxPriority);
2015-06-18 01:40:15 +01:00
Yap_CloseSlots();
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
#endif
2016-05-12 11:44:32 +01:00
static bool bind_variable_names(Term t USES_REGS) {
while (!IsVarTerm(t) && IsPairTerm(t)) {
2015-06-18 01:40:15 +01:00
Term tl = HeadOfTerm(t);
Functor f;
Term tv, t2, t1;
2016-05-12 11:44:32 +01:00
if (!IsApplTerm(tl))
return FALSE;
2015-06-18 01:40:15 +01:00
if ((f = FunctorOfTerm(tl)) != FunctorEq) {
return FALSE;
}
t1 = ArgOfTerm(1, tl);
if (IsVarTerm(t1)) {
2016-05-12 11:44:32 +01:00
Yap_Error(INSTANTIATION_ERROR, t1, "variable_names");
return false;
2015-06-18 01:40:15 +01:00
}
t2 = ArgOfTerm(2, tl);
tv = Yap_MkApplTerm(FunctorDollarVar, 1, &t1);
2015-06-18 01:40:15 +01:00
if (IsVarTerm(t2)) {
2016-04-10 14:21:17 +01:00
YapBind(VarOfTerm(t2), tv);
2015-06-18 01:40:15 +01:00
}
t = TailOfTerm(t);
}
return true;
}
2016-05-12 11:44:32 +01:00
static int unbind_variable_names(Term t USES_REGS) {
while (!IsVarTerm(t) && IsPairTerm(t)) {
2015-06-18 01:40:15 +01:00
Term tl = HeadOfTerm(t);
Functor f;
2016-05-12 11:44:32 +01:00
Term *tp2, t1;
2015-06-18 01:40:15 +01:00
2016-05-12 11:44:32 +01:00
if (!IsApplTerm(tl))
return FALSE;
2015-06-18 01:40:15 +01:00
if ((f = FunctorOfTerm(tl)) != FunctorEq) {
return FALSE;
}
t1 = ArgOfTerm(1, tl);
2016-05-12 11:44:32 +01:00
tp2 = RepAppl(tl) + 2;
2015-06-18 01:40:15 +01:00
while (*tp2 != t1) {
2016-05-12 11:44:32 +01:00
tp2 = (CELL *)*tp2;
2015-06-18 01:40:15 +01:00
}
2016-05-12 11:44:32 +01:00
RESET_VARIABLE(tp2);
2015-06-18 01:40:15 +01:00
t = TailOfTerm(t);
}
return TRUE;
}
2016-05-12 11:44:32 +01:00
static bool write_term(int output_stream, Term t, xarg *args USES_REGS) {
2015-06-18 01:40:15 +01:00
bool rc;
Term cm = CurrentModule;
2016-05-12 11:44:32 +01:00
int depth, prio, flags = 0;
2015-06-18 01:40:15 +01:00
if (args[WRITE_MODULE].used) {
CurrentModule = args[WRITE_MODULE].tvalue;
}
if (args[WRITE_VARIABLE_NAMES].used) {
bind_variable_names(args[WRITE_VARIABLE_NAMES].tvalue PASS_REGS);
flags |= Handle_vars_f;
}
if (args[WRITE_NUMBERVARS].used) {
2015-07-25 03:28:04 +01:00
if (args[WRITE_NUMBERVARS].tvalue == TermTrue)
flags |= Handle_vars_f;
2015-06-18 01:40:15 +01:00
}
if (args[WRITE_ATTRIBUTES].used) {
Term ctl = args[WRITE_ATTRIBUTES].tvalue;
if (ctl == TermWrite) {
flags |= AttVar_None_f;
} else if (ctl == TermPortray) {
2016-05-12 11:44:32 +01:00
flags |= AttVar_None_f | AttVar_Portray_f;
2015-06-18 01:40:15 +01:00
} else if (ctl == TermDots) {
flags |= AttVar_Dots_f;
} else if (ctl != TermIgnore) {
2016-05-12 11:44:32 +01:00
Yap_Error(
2016-11-08 07:37:36 +00:00
DOMAIN_ERROR_WRITE_OPTION, ctl,
2016-05-12 11:44:32 +01:00
"write attributes should be one of {dots,ignore,portray,write}");
2015-06-18 01:40:15 +01:00
rc = false;
goto end;
}
}
2019-02-04 10:42:23 +00:00
if (!args[WRITE_CYCLES].used || (args[WRITE_CYCLES].used
&& args[WRITE_CYCLES].tvalue == TermTrue)) {
flags |= Handle_cyclics_f;
2015-06-18 01:40:15 +01:00
}
2016-05-12 11:44:32 +01:00
if (args[WRITE_QUOTED].used && args[WRITE_QUOTED].tvalue == TermTrue) {
2015-06-18 01:40:15 +01:00
flags |= Quote_illegal_f;
}
if (args[WRITE_IGNORE_OPS].used &&
args[WRITE_IGNORE_OPS].tvalue == TermTrue) {
flags |= Ignore_ops_f;
}
2016-05-12 11:44:32 +01:00
if (args[WRITE_PORTRAY].used && args[WRITE_IGNORE_OPS].tvalue == TermTrue) {
2015-06-18 01:40:15 +01:00
flags |= Ignore_ops_f;
}
2016-05-12 11:44:32 +01:00
if (args[WRITE_PORTRAYED].used && args[WRITE_PORTRAYED].tvalue == TermTrue) {
2015-06-18 01:40:15 +01:00
flags |= Use_portray_f;
}
if (args[WRITE_CHARACTER_ESCAPES].used &&
2016-05-12 11:44:32 +01:00
args[WRITE_CHARACTER_ESCAPES].tvalue == TermFalse) {
2015-06-18 01:40:15 +01:00
flags |= No_Escapes_f;
}
if (args[WRITE_BACKQUOTES].used &&
2016-05-12 11:44:32 +01:00
args[WRITE_BACKQUOTES].tvalue == TermTrue) {
2015-06-18 01:40:15 +01:00
flags |= BackQuote_String_f;
}
if (args[WRITE_BRACE_TERMS].used &&
args[WRITE_BRACE_TERMS].tvalue == TermFalse) {
flags |= No_Brace_Terms_f;
2015-06-18 01:40:15 +01:00
}
2016-05-12 11:44:32 +01:00
if (args[WRITE_FULLSTOP].used && args[WRITE_FULLSTOP].tvalue == TermTrue) {
2015-06-18 01:40:15 +01:00
flags |= Fullstop_f;
}
2016-05-12 11:44:32 +01:00
if (args[WRITE_NL].used && args[WRITE_NL].tvalue == TermTrue) {
2015-06-18 01:40:15 +01:00
flags |= New_Line_f;
}
if (args[WRITE_MAX_DEPTH].used) {
depth = IntegerOfTerm(args[WRITE_MAX_DEPTH].tvalue);
} else
2016-05-12 11:44:32 +01:00
depth = LOCAL_max_depth;
2015-06-18 01:40:15 +01:00
if (args[WRITE_PRIORITY].used) {
prio = IntegerOfTerm(args[WRITE_PRIORITY].tvalue);
} else {
prio = GLOBAL_MaxPriority;
2015-06-18 01:40:15 +01:00
}
2016-05-12 11:44:32 +01:00
Yap_plwrite(t, GLOBAL_Stream + output_stream, depth, flags, prio);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2015-06-18 01:40:15 +01:00
rc = true;
2016-05-12 11:44:32 +01:00
end:
2015-06-18 01:40:15 +01:00
if (args[WRITE_VARIABLE_NAMES].used) {
unbind_variable_names(args[WRITE_VARIABLE_NAMES].tvalue PASS_REGS);
}
CurrentModule = cm;
return rc;
}
2016-07-31 16:17:10 +01:00
/**
*
*/
2016-10-20 04:44:59 +01:00
bool Yap_WriteTerm(int output_stream, Term t, Term opts USES_REGS) {
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(opts, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, opts, NULL);
2015-06-18 01:40:15 +01:00
return false;
}
2018-04-03 18:31:51 +01:00
2015-09-21 23:05:36 +01:00
yhandle_t mySlots = Yap_StartSlots();
2015-07-06 12:03:16 +01:00
LOCK(GLOBAL_Stream[output_stream].streamlock);
2016-07-31 16:17:10 +01:00
write_term(output_stream, t, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
return (TRUE);
2015-06-18 01:40:15 +01:00
}
2016-07-31 16:17:10 +01:00
static Int write_term2(USES_REGS1) {
2015-06-18 01:40:15 +01:00
2016-07-31 16:17:10 +01:00
/* '$write'(+Flags,?Term) */
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
2016-10-20 04:44:59 +01:00
return Yap_WriteTerm(LOCAL_c_output_stream, ARG1, ARG2 PASS_REGS);
2016-07-31 16:17:10 +01:00
}
static Int write_term3(USES_REGS1) {
2016-10-20 04:44:59 +01:00
int output_stream = Yap_CheckTextStream(ARG1, Output_Stream_f, "write/2");
2016-05-14 02:25:51 +01:00
if (output_stream < 0) {
2015-06-18 01:40:15 +01:00
return false;
2016-05-14 02:25:51 +01:00
}
2016-10-20 04:44:59 +01:00
return Yap_WriteTerm(output_stream, ARG2, ARG3 PASS_REGS);
2015-06-18 01:40:15 +01:00
}
2016-05-12 11:44:32 +01:00
static Int write2(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
xarg *args;
yhandle_t mySlots;
2016-05-12 11:44:32 +01:00
int output_stream = Yap_CheckTextStream(ARG1, Output_Stream_f, "write/2");
if (output_stream < 0)
2015-06-18 01:40:15 +01:00
return false;
2018-06-14 11:27:43 +01:00
args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
return false;
}
mySlots = Yap_StartSlots();
2015-07-25 03:28:04 +01:00
args[WRITE_NUMBERVARS].used = true;
args[WRITE_NUMBERVARS].tvalue = TermTrue;
2016-05-12 11:44:32 +01:00
write_term(output_stream, ARG2, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
2016-05-12 11:44:32 +01:00
static Int write1(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
int output_stream = LOCAL_c_output_stream;
2016-05-12 11:44:32 +01:00
if (output_stream == -1)
output_stream = 1;
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
2016-05-12 11:44:32 +01:00
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
2015-06-18 01:40:15 +01:00
return false;
}
2015-09-21 23:05:36 +01:00
yhandle_t mySlots = Yap_StartSlots();
args[WRITE_NUMBERVARS].used = true;
2015-09-21 23:05:36 +01:00
args[WRITE_NUMBERVARS].tvalue = TermTrue;
2015-07-06 12:03:16 +01:00
LOCK(GLOBAL_Stream[output_stream].streamlock);
2016-05-12 11:44:32 +01:00
write_term(output_stream, ARG1, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
2016-05-12 11:44:32 +01:00
static Int write_canonical1(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
int output_stream = LOCAL_c_output_stream;
2016-05-12 11:44:32 +01:00
if (output_stream == -1)
output_stream = 1;
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
2015-06-18 01:40:15 +01:00
return false;
}
2015-09-21 23:05:36 +01:00
yhandle_t mySlots = Yap_StartSlots();
2015-06-18 01:40:15 +01:00
args[WRITE_IGNORE_OPS].used = true;
args[WRITE_IGNORE_OPS].tvalue = TermTrue;
args[WRITE_QUOTED].used = true;
args[WRITE_QUOTED].tvalue = TermTrue;
2015-07-06 12:03:16 +01:00
LOCK(GLOBAL_Stream[output_stream].streamlock);
2016-05-12 11:44:32 +01:00
write_term(output_stream, ARG1, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
2016-05-12 11:44:32 +01:00
static Int write_canonical(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
return false;
}
2016-05-12 11:44:32 +01:00
int output_stream = Yap_CheckTextStream(ARG1, Output_Stream_f, "write/2");
2016-05-14 02:25:51 +01:00
if (output_stream < 0) {
2016-10-20 04:44:59 +01:00
free(args);
2015-06-18 01:40:15 +01:00
return false;
2016-05-14 02:25:51 +01:00
}
2015-09-21 23:05:36 +01:00
yhandle_t mySlots = Yap_StartSlots();
2015-06-18 01:40:15 +01:00
args[WRITE_IGNORE_OPS].used = true;
args[WRITE_IGNORE_OPS].tvalue = TermTrue;
args[WRITE_QUOTED].used = true;
args[WRITE_QUOTED].tvalue = TermTrue;
2016-05-12 11:44:32 +01:00
write_term(output_stream, ARG2, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
2016-05-12 11:44:32 +01:00
static Int writeq1(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
2016-05-12 11:44:32 +01:00
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
2015-09-21 23:05:36 +01:00
return false;
}
2015-06-18 01:40:15 +01:00
yhandle_t mySlots = Yap_StartSlots();
int output_stream = LOCAL_c_output_stream;
2016-05-14 02:25:51 +01:00
if (output_stream == -1) {
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
output_stream = 1;
2016-05-14 02:25:51 +01:00
}
2015-06-18 01:40:15 +01:00
args[WRITE_NUMBERVARS].used = true;
args[WRITE_NUMBERVARS].tvalue = TermTrue;
args[WRITE_QUOTED].used = true;
args[WRITE_QUOTED].tvalue = TermTrue;
2016-05-12 11:44:32 +01:00
write_term(output_stream, ARG1, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
2016-05-12 11:44:32 +01:00
static Int writeq(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
return false;
}
2016-05-12 11:44:32 +01:00
int output_stream = Yap_CheckTextStream(ARG1, Output_Stream_f, "write/2");
2016-05-14 02:25:51 +01:00
if (output_stream < 0) {
2016-10-20 04:44:59 +01:00
free(args);
2015-06-18 01:40:15 +01:00
return false;
2016-05-14 02:25:51 +01:00
}
2015-09-21 23:05:36 +01:00
yhandle_t mySlots = Yap_StartSlots();
2016-05-12 11:44:32 +01:00
args[WRITE_NUMBERVARS].used = true;
args[WRITE_NUMBERVARS].tvalue = TermTrue;
args[WRITE_QUOTED].used = true;
args[WRITE_QUOTED].tvalue = TermTrue;
write_term(output_stream, ARG2, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
2016-05-12 11:44:32 +01:00
static Int print1(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
2016-05-12 11:44:32 +01:00
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
2015-06-18 01:40:15 +01:00
return false;
}
2015-09-21 23:05:36 +01:00
yhandle_t mySlots = Yap_StartSlots();
int output_stream = LOCAL_c_output_stream;
2016-05-14 02:25:51 +01:00
if (output_stream == -1) {
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
output_stream = 1;
2016-05-14 02:25:51 +01:00
}
2016-05-12 11:44:32 +01:00
args[WRITE_PORTRAY].used = true;
args[WRITE_PORTRAY].tvalue = TermTrue;
args[WRITE_NUMBERVARS].used = true;
args[WRITE_NUMBERVARS].tvalue = TermTrue;
2015-07-06 12:03:16 +01:00
LOCK(GLOBAL_Stream[output_stream].streamlock);
2016-05-12 11:44:32 +01:00
write_term(output_stream, ARG1, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
2016-05-12 11:44:32 +01:00
static Int print(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
2016-05-12 11:44:32 +01:00
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
return false;
}
2016-05-12 11:44:32 +01:00
int output_stream = Yap_CheckTextStream(ARG1, Output_Stream_f, "write/2");
2016-05-14 02:25:51 +01:00
if (output_stream < 0) {
2016-10-20 04:44:59 +01:00
free(args);
2015-06-18 01:40:15 +01:00
return false;
2016-05-14 02:25:51 +01:00
}
2015-09-21 23:05:36 +01:00
yhandle_t mySlots = Yap_StartSlots();
args[WRITE_PORTRAY].used = true;
args[WRITE_PORTRAY].tvalue = TermTrue;
2015-06-18 01:40:15 +01:00
args[WRITE_NUMBERVARS].used = true;
2015-09-21 23:05:36 +01:00
args[WRITE_NUMBERVARS].tvalue = TermTrue;
2016-05-12 11:44:32 +01:00
write_term(output_stream, ARG2, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2016-05-12 11:44:32 +01:00
return (TRUE);
2015-06-18 01:40:15 +01:00
}
2016-05-12 11:44:32 +01:00
static Int writeln1(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
int output_stream = LOCAL_c_output_stream;
2016-05-12 11:44:32 +01:00
if (output_stream == -1)
output_stream = 1;
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
2015-06-18 01:40:15 +01:00
return false;
}
2015-09-21 23:05:36 +01:00
yhandle_t mySlots = Yap_StartSlots();
2015-07-25 03:28:04 +01:00
args[WRITE_NL].used = true;
2015-06-18 01:40:15 +01:00
args[WRITE_NL].tvalue = TermTrue;
2015-07-25 03:28:04 +01:00
args[WRITE_NUMBERVARS].used = true;
args[WRITE_NUMBERVARS].tvalue = TermTrue;
2019-01-24 13:27:23 +00:00
args[WRITE_CYCLES].used = true;
args[WRITE_CYCLES].tvalue = TermTrue;
2015-07-06 12:03:16 +01:00
LOCK(GLOBAL_Stream[output_stream].streamlock);
2016-05-12 11:44:32 +01:00
write_term(output_stream, ARG1, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
2016-05-12 11:44:32 +01:00
static Int writeln(USES_REGS1) {
2015-06-18 01:40:15 +01:00
/* notice: we must have ASP well set when using portray, otherwise
we cannot make recursive Prolog calls */
2018-06-14 11:27:43 +01:00
xarg *args = Yap_ArgListToVector(TermNil, write_defs, WRITE_END,
DOMAIN_ERROR_WRITE_OPTION);
if (args == NULL) {
if (LOCAL_Error_TYPE)
2016-10-20 04:44:59 +01:00
Yap_Error(LOCAL_Error_TYPE, TermNil, NULL);
2015-07-06 12:03:16 +01:00
return false;
}
2016-05-12 11:44:32 +01:00
int output_stream = Yap_CheckTextStream(ARG1, Output_Stream_f, "writeln/2");
2016-05-14 02:25:51 +01:00
if (output_stream < 0) {
2016-10-20 04:44:59 +01:00
free(args);
return false;
2016-05-14 02:25:51 +01:00
}
2015-09-21 23:05:36 +01:00
yhandle_t mySlots = Yap_StartSlots();
2015-06-18 01:40:15 +01:00
args[WRITE_NL].used = true;
args[WRITE_NL].tvalue = TermTrue;
2015-07-25 03:28:04 +01:00
args[WRITE_NUMBERVARS].used = true;
args[WRITE_NUMBERVARS].tvalue = TermTrue;
2019-01-24 13:27:23 +00:00
args[WRITE_CYCLES].used = true;
args[WRITE_CYCLES].tvalue = TermTrue;
2016-05-12 11:44:32 +01:00
write_term(output_stream, ARG2, args PASS_REGS);
2015-07-06 12:03:16 +01:00
UNLOCK(GLOBAL_Stream[output_stream].streamlock);
2016-10-20 04:44:59 +01:00
free(args);
2016-05-12 11:44:32 +01:00
Yap_CloseSlots(mySlots);
2016-03-29 01:57:55 +01:00
Yap_RaiseException();
2015-06-18 01:40:15 +01:00
return (TRUE);
}
2016-05-12 11:44:32 +01:00
static Int p_write_depth(USES_REGS1) { /* write_depth(Old,New) */
Term t1 = Deref(ARG1);
Term t2 = Deref(ARG2);
Term t3 = Deref(ARG3);
2015-06-18 01:40:15 +01:00
2016-05-12 11:44:32 +01:00
if (!IsVarTerm(t1) && !IsIntegerTerm(t1)) {
Yap_Error(TYPE_ERROR_INTEGER, t1, "write_depth/3");
2015-06-18 01:40:15 +01:00
return FALSE;
}
2016-05-12 11:44:32 +01:00
if (!IsVarTerm(t2) && !IsIntegerTerm(t2)) {
Yap_Error(TYPE_ERROR_INTEGER, t2, "write_depth/3");
2015-06-18 01:40:15 +01:00
return FALSE;
}
2016-05-12 11:44:32 +01:00
if (!IsVarTerm(t3) && !IsIntegerTerm(t3)) {
Yap_Error(TYPE_ERROR_INTEGER, t3, "write_depth/3");
2015-06-18 01:40:15 +01:00
return FALSE;
}
2016-05-12 11:44:32 +01:00
if (IsVarTerm(t1)) {
Term t = MkIntegerTerm(LOCAL_max_depth);
if (!Yap_unify_constant(t1, t))
return FALSE;
} else
LOCAL_max_depth = IntegerOfTerm(t1);
if (IsVarTerm(t2)) {
Term t = MkIntegerTerm(LOCAL_max_list);
if (!Yap_unify_constant(t2, t))
return FALSE;
} else
LOCAL_max_list = IntegerOfTerm(t2);
if (IsVarTerm(t3)) {
Term t = MkIntegerTerm(LOCAL_max_write_args);
if (!Yap_unify_constant(t3, t))
return FALSE;
} else
LOCAL_max_write_args = IntegerOfTerm(t3);
2015-06-18 01:40:15 +01:00
return TRUE;
}
2016-05-12 11:44:32 +01:00
static Int dollar_var(USES_REGS1) {
2015-06-18 01:40:15 +01:00
Term in = Deref(ARG1);
2015-10-27 23:07:11 +00:00
if (IsVarTerm(in)) {
Term t2;
2016-05-12 11:44:32 +01:00
if (!IsVarTerm(t2 = Deref(ARG2))) {
if (IsApplTerm(t2) && FunctorOfTerm(t2) == FunctorDollarVar) {
2015-10-27 23:07:11 +00:00
return Yap_unify(ArgOfTerm(1, t2), ARG1);
}
2016-05-12 11:44:32 +01:00
Yap_Error(TYPE_ERROR_COMPOUND, ARG2, "");
2015-10-27 23:07:11 +00:00
return false;
} else {
2016-05-12 11:44:32 +01:00
Yap_Error(INSTANTIATION_ERROR, ARG2, "");
2015-10-27 23:07:11 +00:00
}
}
2016-05-12 11:44:32 +01:00
Term t2 = Yap_unify(MkVarTerm(), ARG1);
Term tv = Yap_MkApplTerm(FunctorDollarVar, 1, &t2);
2015-10-22 12:01:54 +01:00
return Yap_unify(tv, ARG2);
}
2016-07-31 16:17:10 +01:00
static Int term_to_string(USES_REGS1) {
Term t2 = Deref(ARG2), rc = false, t1 = Deref(ARG1);
const char *s;
if (IsVarTerm(t2)) {
2018-07-10 23:21:19 +01:00
s = Yap_TermToBuffer(t1, Quote_illegal_f | Handle_vars_f);
2016-07-31 16:17:10 +01:00
if (!s || !MkStringTerm(s)) {
Yap_Error(RESOURCE_ERROR_HEAP, t1,
"Could not get memory from the operating system");
return false;
}
return Yap_unify(ARG2, MkStringTerm(s));
} else if (!IsStringTerm(t2)) {
2018-12-11 12:50:52 +00:00
Yap_Error(TYPE_ERROR_STRING, t2, "term_to_string/3");
2016-07-31 16:17:10 +01:00
return false;
} else {
s = StringOfTerm(t2);
}
2018-12-11 12:50:52 +00:00
yhandle_t y1 = Yap_InitHandle( t1 );
return (rc = readFromBuffer(s, TermNil)) != 0L && Yap_unify(rc, Yap_PopHandle(y1));
2016-07-31 16:17:10 +01:00
}
static Int term_to_atom(USES_REGS1) {
Term t2 = Deref(ARG2), ctl, rc = false;
Atom at;
if (IsVarTerm(t2)) {
2018-07-21 12:08:09 +01:00
const char *s =
Yap_TermToBuffer(Deref(ARG1), Quote_illegal_f | Handle_vars_f);
2016-07-31 16:17:10 +01:00
if (!s || !(at = Yap_UTF8ToAtom((const unsigned char *)s))) {
Yap_Error(RESOURCE_ERROR_HEAP, t2,
"Could not get memory from the operating system");
return false;
}
return Yap_unify(ARG2, MkAtomTerm(at));
} else if (!IsAtomTerm(t2)) {
Yap_Error(TYPE_ERROR_ATOM, t2, "atom_to_term/2");
return (FALSE);
} else {
at = AtomOfTerm(t2);
}
ctl = TermNil;
2018-04-02 14:49:45 +01:00
return ((rc = Yap_UBufferToTerm(RepAtom(at)->UStrOfAE, ctl))) &&
2016-12-04 18:52:42 +00:00
Yap_unify(rc, ARG1);
2016-07-31 16:17:10 +01:00
}
2018-07-21 12:08:09 +01:00
char *Yap_TermToBuffer(Term t, int flags) {
2018-07-13 12:27:58 +01:00
CACHE_REGS
2018-07-21 12:08:09 +01:00
int sno = Yap_open_buf_write_stream(LOCAL_encoding, flags);
2018-07-13 12:27:58 +01:00
if (sno < 0)
return NULL;
if (t == 0)
return NULL;
else
t = Deref(t);
GLOBAL_Stream[sno].encoding = LOCAL_encoding;
GLOBAL_Stream[sno].status |= CloseOnException_Stream_f;
2019-05-01 01:34:58 +01:00
GLOBAL_Stream[sno].status &= ~FreeOnClose_Stream_f;
2018-07-13 12:27:58 +01:00
Yap_plwrite(t, GLOBAL_Stream + sno, 0, flags, GLOBAL_MaxPriority);
char *new = Yap_MemExportStreamPtr(sno);
Yap_CloseStream(sno);
return new;
}
2016-05-12 11:44:32 +01:00
void Yap_InitWriteTPreds(void) {
Yap_InitCPred("write_term", 2, write_term2, SyncPredFlag);
Yap_InitCPred("write_term", 3, write_term3, SyncPredFlag);
Yap_InitCPred("write", 1, write1, SyncPredFlag);
Yap_InitCPred("write", 2, write2, SyncPredFlag);
Yap_InitCPred("writeq", 1, writeq1, SyncPredFlag);
Yap_InitCPred("writeq", 2, writeq, SyncPredFlag);
Yap_InitCPred("writeln", 1, writeln1, SyncPredFlag);
Yap_InitCPred("writeln", 2, writeln, SyncPredFlag);
Yap_InitCPred("write_canonical", 1, write_canonical1, SyncPredFlag);
Yap_InitCPred("write_canonical", 2, write_canonical, SyncPredFlag);
Yap_InitCPred("print", 1, print1, SyncPredFlag);
Yap_InitCPred("print", 2, print, SyncPredFlag);
Yap_InitCPred("write_depth", 3, p_write_depth, SafePredFlag | SyncPredFlag);
;
2016-10-20 04:44:59 +01:00
Yap_InitCPred("term_to_string", 3, term_to_string, 0);
Yap_InitCPred("term_to_atom", 3, term_to_atom, 0);
2016-07-31 16:17:10 +01:00
Yap_InitCPred("write_depth", 3, p_write_depth, SafePredFlag | SyncPredFlag);
;
2016-05-12 11:44:32 +01:00
Yap_InitCPred("$VAR", 2, dollar_var, SafePredFlag);
;
2015-06-18 01:40:15 +01:00
}