/************************************************************************* * * * 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 * * * *************************************************************************/ #ifdef SCCS static char SccsId[] = "%W% %G%"; #endif /** * @file iopreds.c * @author VITOR SANTOS COSTA * @date Wed Jan 20 00:45:56 2016 * * @brief main open and close predicates over generic streams. * */ /* * This file includes the definition of a miscellania of standard predicates * *for yap refering to: Files and GLOBAL_1588 *ams, Simple Input/Output, * */ #include "Yap.h" #include "YapEval.h" #include "YapHeap.h" #include "YapText.h" #include "Yatom.h" #include "yapio.h" #include #if HAVE_UNISTD_H #include #endif #if HAVE_STDARG_H #include #endif #if HAVE_CTYPE_H #endif #if HAVE_WCTYPE_H #endif #if HAVE_SYS_TIME_H #include #endif #if HAVE_SYS_TYPES_H #include #endif #ifdef HAVE_SYS_STAT_H #include #endif #if HAVE_SYS_SELECT_H && !_MSC_VER && !defined(__MINGW32__) #include #endif #ifdef HAVE_UNISTD_H #endif #if HAVE_STRING_H #include #endif #if HAVE_SIGNAL_H #include #endif #if HAVE_FCNTL_H /* for O_BINARY and O_TEXT in WIN32 */ #include #endif #ifdef _WIN32 #if HAVE_IO_H /* Windows */ #include #endif #endif #if !HAVE_STRNCAT #define strncat(X, Y, Z) strcat(X, Y) #endif #if !HAVE_STRNCPY #define strncpy(X, Y, Z) strcpy(X, Y) #endif #if _MSC_VER || defined(__MINGW32__) #if HAVE_SOCKET #include #endif #include #ifndef S_ISDIR #define S_ISDIR(x) (((x)&_S_IFDIR) == _S_IFDIR) #endif #endif #include "iopreds.h" #include "getw.h" static int get_wchar_from_file(int); FILE *Yap_stdin; FILE *Yap_stdout; FILE *Yap_stderr; static Term gethdir(Term t) { CACHE_REGS Atom aref = AtomOfTerm(t); char *s = RepAtom(aref)->StrOfAE; size_t nsz; s = strncpy(LOCAL_FileNameBuf, RepAtom(aref)->StrOfAE, MAXPATHLEN - 1); if (!s) { return false; } if (TermDot == t) { return TermEmptyAtom; } nsz = strlen(s); if (!Yap_dir_separator(s[nsz - 1])) { #if _WIN32 s[nsz] = '\\'; #else s[nsz] = '/'; #endif s[nsz + 1] = '\0'; } return MkAtomTerm(Yap_LookupAtom(s)); } static Term issolutions(Term t) { if (t == TermFirst || t == TermAll) return t; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "solutions in {first, all}."); return TermZERO; } if (IsAtomTerm(t)) { Yap_Error(DOMAIN_ERROR_SOLUTIONS, t, "solutions in {first, all}"); return TermZERO; } Yap_Error(TYPE_ERROR_ATOM, t, "solutions in {first, all}}"); return TermZERO; } static Term is_file_type(Term t) { if (t == TermTxt || t == TermProlog || t == TermSource || t == TermExecutable || t == TermQly || t == TermDirectory) return t; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "file_type in {txt,prolog,exe,directory...}"); return TermZERO; } if (IsAtomTerm(t)) { Yap_Error(DOMAIN_ERROR_FILE_TYPE, t, "file_type in {txt,prolog,exe,directory...}"); return TermZERO; } Yap_Error(TYPE_ERROR_ATOM, t, "file_type in {txt,prolog,exe,directory...}"); return TermZERO; } static Term is_file_errors(Term t) { if (t == TermFail || t == TermError) return t; if (IsVarTerm(t)) { Yap_Error(INSTANTIATION_ERROR, t, "file_error in {fail,error}."); return TermZERO; } if (IsAtomTerm(t)) { Yap_Error(DOMAIN_ERROR_FILE_ERRORS, t, "file_error in {fail,error}."); return TermZERO; } Yap_Error(TYPE_ERROR_ATOM, t, "file_error in {fail,error}."); return TermZERO; } int ResetEOF(StreamDesc *s) { if (s->status & Eof_Error_Stream_f) { Atom name = s->name; // Yap_CloseStream(s - GLOBAL_Stream); Yap_Error(PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM, MkAtomTerm(name), "GetC"); return FALSE; } else if (s->status & Reset_Eof_Stream_f) { s->status &= ~Push_Eof_Stream_f; /* reset the eof indicator on file */ if (feof(s->file)) clearerr(s->file); /* reset our function for reading input */ Yap_DefaultStreamOps(s); /* next, reset our own error indicator */ s->status &= ~Eof_Stream_f; /* try reading again */ return TRUE; } else { s->status |= Past_Eof_Stream_f; return FALSE; } } /* handle reading from a stream after having found an EOF */ static int EOFWGetc(int sno) { register StreamDesc *s = &GLOBAL_Stream[sno]; if (s->status & Push_Eof_Stream_f) { /* ok, we have pushed an EOF, send it away */ s->status &= ~Push_Eof_Stream_f; return EOF; } if (ResetEOF(s)) { Yap_DefaultStreamOps(s); return (s->stream_wgetc(sno)); } return EOF; } static int EOFGetc(int sno) { register StreamDesc *s = &GLOBAL_Stream[sno]; if (s->status & Push_Eof_Stream_f) { /* ok, we have pushed an EOF, send it away */ s->status &= ~Push_Eof_Stream_f; ResetEOF(s); return EOF; } if (ResetEOF(s)) { Yap_DefaultStreamOps(s); return s->stream_getc(sno); } return EOF; } static void unix_upd_stream_info(StreamDesc *s) { if (s->status & InMemory_Stream_f) { s->status |= Seekable_Stream_f; return; } Yap_socketStream(s); #if _MSC_VER || defined(__MINGW32__) { if (_isatty(_fileno(s->file))) { s->status |= Tty_Stream_f | Reset_Eof_Stream_f | Promptable_Stream_f; /* make all console descriptors unbuffered */ setvbuf(s->file, NULL, _IONBF, 0); return; } #if _MSC_VER /* standard error stream should never be buffered */ else if (StdErrStream == s - GLOBAL_Stream) { setvbuf(s->file, NULL, _IONBF, 0); } #endif s->status |= Seekable_Stream_f; return; } #else #if HAVE_ISATTY #if __simplescalar__ /* isatty does not seem to work with simplescar. I'll assume the first three streams will probably be ttys (pipes are not thatg different) */ if (s - Stream < 3) { s->name = AtomTty; s->status |= Tty_Stream_f | Reset_Eof_Stream_f | Promptable_Stream_f; } #else { int filedes; /* visualc */ if (!s->file) { s->name = AtomNil; return; } filedes = fileno(s->file); if (isatty(filedes)) { #if HAVE_TTYNAME int rc = ttyname_r(filedes, LOCAL_FileNameBuf, YAP_FILENAME_MAX - 1); if (rc == 0) s->name = Yap_LookupAtom(LOCAL_FileNameBuf); else s->name = AtomTtys; #else s->name = AtomTty; #endif s->status |= Tty_Stream_f | Reset_Eof_Stream_f | Promptable_Stream_f; return; } } #endif #endif /* HAVE_ISATTY */ #endif /* _MSC_VER */ s->status |= Seekable_Stream_f; } static void default_peek(StreamDesc *st) { { st->stream_peek = Yap_peekChar; st->stream_wpeek = Yap_peekWide; } if (st->status & Eof_Stream_f) { st->stream_peek = EOFPeek; st->stream_wpeek = EOFPeek; st->stream_getc = EOFGetc; st->stream_wgetc = EOFWGetc; } if (GLOBAL_CharConversionTable != NULL) st->stream_wgetc_for_read = ISOWGetc; else st->stream_wgetc_for_read = st->stream_wgetc; } void Yap_DefaultStreamOps(StreamDesc *st) { CACHE_REGS st->stream_wputc = put_wchar; st->stream_wgetc = get_wchar; if (st->vfs && !st->file) { st->stream_putc = st->vfs->put_char; st->stream_wputc = st->vfs->put_char; st->stream_getc = st->vfs->get_char; st->stream_wgetc = st->vfs->get_char; default_peek(st); return; } else { if (st->encoding == ENC_ISO_UTF8) st->stream_wgetc = get_wchar_UTF8; st->stream_putc = FilePutc; st->stream_getc = PlGetc; if (st->status & Pipe_Stream_f) { Yap_PipeOps(st); } else if (st->status & InMemory_Stream_f) { Yap_MemOps(st); } else if (st->status & Tty_Stream_f) { Yap_ConsoleOps(st); } else { unix_upd_stream_info(st); } if (st->status & (Promptable_Stream_f)) { Yap_ConsoleOps(st); } #ifndef _WIN32 else if (st->file != NULL && 0 && !(st->status & InMemory_Stream_f)) { st->stream_wgetc = get_wchar_from_file; } #endif if (st->buf.on) { st->stream_getc = Yap_popChar; st->stream_wgetc = Yap_popChar; } } #if USE_READLINE if (st->status & Readline_Stream_f) { st->stream_peek = Yap_ReadlinePeekChar; st->stream_wpeek = Yap_ReadlinePeekChar; } /* else { st->stream_peek = Yap_peekWithGetc; st->stream_wpeek = Yap_peekWideWithGetwc; } } else if (st->status & Seekable_Stream_f) { st->stream_peek = Yap_peekWithSeek; st->stream_wpeek = Yap_peekWideWithSeek; } */ else #endif default_peek(st); } static void InitFileIO(StreamDesc *s) { CACHE_REGS Yap_DefaultStreamOps(s); } static void InitStdStream(int sno, SMALLUNSGN flags, FILE *file, VFS_t *vfsp) { StreamDesc *s = &GLOBAL_Stream[sno]; s->file = file; s->status = flags; s->linepos = 0; s->linecount = 1; s->charcount = 0; s->vfs = vfsp; s->buf.on = false; s->encoding = ENC_ISO_UTF8; INIT_LOCK(s->streamlock); if (vfsp == NULL) { unix_upd_stream_info(s); } /* Getting streams to prompt is a mess because we need for cooperation between readers and writers to the stream :-( */ InitFileIO(s); switch (sno) { case 0: s->name = AtomUserIn; break; case 1: s->name = AtomUserOut; break; default: s->name = AtomUserErr; break; } s->user_name = MkAtomTerm(s->name); #if LIGHT s->status |= Tty_Stream_f | Promptable_Stream_f; #endif s->buf.on = false; Yap_DefaultStreamOps(s); #if HAVE_SETBUF if (s->status & Tty_Stream_f && sno == 0) { /* make sure input is unbuffered if it comes from stdin, this makes life simpler for interrupt handling */ setbuf(stdin, NULL); // fprintf(stderr,"here I am\n"); } #endif /* HAVE_SETBUF */ } void Yap_InitStdStream(int sno, unsigned int flags, FILE *file, VFS_t *vfsp) { InitStdStream(sno, flags, file, vfsp); } Term Yap_StreamUserName(int sno) { Term atname; StreamDesc *s = &GLOBAL_Stream[sno]; if (s->user_name != 0L) { return (s->user_name); } if ((atname = StreamName(sno))) return atname; return TermNil; } static void InitStdStreams(void) { CACHE_REGS if (LOCAL_sockets_io) { InitStdStream(StdInStream, Input_Stream_f, NULL, NULL); InitStdStream(StdOutStream, Output_Stream_f, NULL, NULL); InitStdStream(StdErrStream, Output_Stream_f, NULL, NULL); } else { InitStdStream(StdInStream, Input_Stream_f, stdin, NULL); InitStdStream(StdOutStream, Output_Stream_f, stdout, NULL); InitStdStream(StdErrStream, Output_Stream_f, stderr, NULL); } GLOBAL_Stream[StdInStream].name = Yap_LookupAtom("user_input"); GLOBAL_Stream[StdOutStream].name = Yap_LookupAtom("user_output"); GLOBAL_Stream[StdErrStream].name = Yap_LookupAtom("user_error"); #if USE_READLINE if (GLOBAL_Stream[StdInStream].status & Tty_Stream_f && GLOBAL_Stream[StdOutStream].status & Tty_Stream_f && GLOBAL_Stream[StdErrStream].status & Tty_Stream_f && !Yap_Embedded) { Yap_InitReadline(TermTrue); } #endif LOCAL_c_input_stream = StdInStream; LOCAL_c_output_stream = StdOutStream; LOCAL_c_error_stream = StdErrStream; } void Yap_InitStdStreams(void) { InitStdStreams(); } Int PlIOError__(const char *file, const char *function, int lineno, yap_error_number type, Term culprit, ...) { if (trueLocalPrologFlag(FILEERRORS_FLAG) || type == RESOURCE_ERROR_MAX_STREAMS /* do not catch resource errors */) { va_list args; const char *format; char who[1024]; va_start(args, culprit); format = va_arg(args, char *); if (format) { vsnprintf(who, 1023, format, args); } else { who[0] = '\0'; } va_end(args); Yap_Error__(false, file, function, lineno, type, culprit, who); /* and fail */ return false; } else { pop_text_stack(0); memset(LOCAL_ActiveError, 0, sizeof(*LOCAL_ActiveError)); return false; } } static int eolflg = 1; static char my_line[200] = {0}; static char *lp = my_line; FILE *curfile, *Yap_logfile; bool Yap_Option[256]; #ifdef MACC static void InTTYLine(char *line) { char *p = line; char ch; while ((ch = InKey()) != '\n' && ch != '\r') if (ch == 8) { if (line < p) BackupTTY(*--p); } else TTYChar(*p++ = ch); TTYChar('\n'); *p = 0; } #endif void Yap_DebugSetIFile(char *fname) { if (curfile) fclose(curfile); curfile = fopen(fname, "r"); if (curfile == NULL) { curfile = stdin; Yap_Warning("%% YAP open %s for input\n", fname); } } void Yap_DebugEndline() { *lp = 0; } int Yap_DebugGetc() { int ch; if (eolflg) { if (curfile != NULL) { if (fgets(my_line, 200, curfile) == 0) curfile = NULL; } if (curfile == NULL) if (fgets(my_line, 200, stdin) == NULL) { return EOF; } eolflg = 0; lp = my_line; } if ((ch = *lp++) == 0) ch = '\n', eolflg = 1; if (Yap_Option['l' - 96]) putc(ch, Yap_logfile); return (ch); } int Yap_DebugPutc(FILE *s, wchar_t ch) { if (Yap_Option['l' - 96]) (void)putc(ch, Yap_logfile); return (putc(ch, stderr)); } int Yap_DebugPuts(FILE *s, const char *sch) { if (Yap_Option['l' - 96]) (void)fputs(sch, Yap_logfile); return fputs(sch, stderr); } void Yap_DebugErrorPuts(const char *s) { Yap_DebugPuts(stderr, s); } void Yap_DebugPlWrite(Term t) { if (t == 0) fprintf(stderr, "NULL"); Yap_plwrite(t, GLOBAL_Stream + 2, 0, 0, GLOBAL_MaxPriority); } void Yap_DebugPlWriteln(Term t) { CACHE_REGS if (t == 0) fprintf(stderr, "NULL"); Yap_plwrite(t, NULL, 15, 0, GLOBAL_MaxPriority); Yap_DebugPutc(GLOBAL_Stream[LOCAL_c_error_stream].file, '.'); Yap_DebugPutc(GLOBAL_Stream[LOCAL_c_error_stream].file, 10); } void Yap_DebugErrorPutc(int c) { CACHE_REGS Yap_DebugPutc(GLOBAL_Stream[LOCAL_c_error_stream].file, c); } void Yap_DebugWriteIndicator(PredEntry *ap) { CACHE_REGS Term tmod = ap->ModuleOfPred; if (!tmod) tmod = TermProlog; #if THREADS Yap_DebugPlWrite(MkIntegerTerm(worker_id)); Yap_DebugPutc(stderr, ' '); #endif Yap_DebugPutc(stderr, '>'); Yap_DebugPutc(stderr, '\t'); Yap_DebugPlWrite(tmod); Yap_DebugPutc(stderr, ':'); if (ap->ModuleOfPred == IDB_MODULE) { Term t = Deref(ARG1); if (IsAtomTerm(t)) { Yap_DebugPlWrite(t); } else if (IsIntegerTerm(t)) { Yap_DebugPlWrite(t); } else { Functor f = FunctorOfTerm(t); Atom At = NameOfFunctor(f); Yap_DebugPlWrite(MkAtomTerm(At)); Yap_DebugPutc(stderr, '/'); Yap_DebugPlWrite(MkIntegerTerm(ArityOfFunctor(f))); } } else { if (ap->ArityOfPE == 0) { Atom At = (Atom)ap->FunctorOfPred; Yap_DebugPlWrite(MkAtomTerm(At)); } else { Functor f = ap->FunctorOfPred; Atom At = NameOfFunctor(f); Yap_DebugPlWrite(MkAtomTerm(At)); Yap_DebugPutc(stderr, '/'); Yap_DebugPlWrite(MkIntegerTerm(ArityOfFunctor(f))); } } Yap_DebugPutc(stderr, '\n'); } /* static */ int FilePutc(int sno, int ch) { StreamDesc *s = &GLOBAL_Stream[sno]; #if MAC || _MSC_VER if (ch == 10) { ch = '\n'; } #endif putc(ch, s->file); #if MAC || _MSC_VER if (ch == 10) { fflush(s->file); } #endif count_output_char(ch, s); return ((int)ch); } static int NullPutc(int sno, int ch) { StreamDesc *s = &GLOBAL_Stream[sno]; #if MAC || _MSC_VER if (ch == 10) { ch = '\n'; } #endif count_output_char(ch, s); return ((int)ch); } /* check if we read a LOCAL_newline or an EOF */ int console_post_process_eof(StreamDesc *s) { CACHE_REGS if (!ResetEOF(s)) { s->status |= Eof_Stream_f; s->stream_getc = EOFGetc; s->stream_wgetc = EOFWGetc; s->stream_wgetc_for_read = EOFWGetc; LOCAL_newline = true; } return EOFCHAR; } /* check if we read a newline or an EOF */ int post_process_read_wchar(int ch, size_t n, StreamDesc *s) { if (ch == EOF) { return post_process_weof(s); } #if DEBUG if (GLOBAL_Option[1]) { static int v; fprintf(stderr, "%d %C\n", v, ch); v++; } #endif s->charcount += n; s->linepos += n; if (ch == '\n') { ++s->linecount; s->linepos = 0; /* don't convert if the stream is binary */ if (!(s->status & Binary_Stream_f)) ch = 10; } return ch; } int post_process_weof(StreamDesc *s) { if (!ResetEOF(s)) { s->status |= Eof_Stream_f; s->stream_wgetc = EOFWGetc; s->stream_getc = EOFGetc; s->stream_wgetc_for_read = EOFWGetc; } return EOFCHAR; } void *Yap_RepStreamFromId(int sno) { return GLOBAL_Stream + (sno); } /** * caled after EOF found a peek, it just calls console_post_process to *conclude *the job. * * @param sno * * @return EOF */ int EOFPeek(int sno) { return EOFCHAR; } int EOFWPeek(int sno) { return EOFCHAR; } /* standard routine, it should read from anything pointed by a FILE *. It could be made more efficient by doing our own buffering and avoiding post_process_read_char, something to think about */ int PlGetc(int sno) { StreamDesc *s = &GLOBAL_Stream[sno]; return fgetc(s->file); } // layered version static inline int get_wchar_from_file(int sno) { return post_process_read_wchar(fgetwc(GLOBAL_Stream[sno].file), 1, GLOBAL_Stream + sno); } #ifndef MB_LEN_MAX #define MB_LEN_MAX 6 #endif static int handle_write_encoding_error(int sno, wchar_t ch) { if (GLOBAL_Stream[sno].status & RepError_Xml_f) { /* use HTML/XML encoding in ASCII */ int i = ch, digits = 1; GLOBAL_Stream[sno].stream_putc(sno, '&'); GLOBAL_Stream[sno].stream_putc(sno, '#'); while (digits < i) digits *= 10; if (digits > i) digits /= 10; while (i) { GLOBAL_Stream[sno].stream_putc(sno, i / digits); i %= 10; digits /= 10; } GLOBAL_Stream[sno].stream_putc(sno, ';'); return ch; } else if (GLOBAL_Stream[sno].status & RepError_Prolog_f) { /* write quoted */ GLOBAL_Stream[sno].stream_putc(sno, '\\'); GLOBAL_Stream[sno].stream_putc(sno, 'u'); GLOBAL_Stream[sno].stream_putc(sno, ch >> 24); GLOBAL_Stream[sno].stream_putc(sno, 256 & (ch >> 16)); GLOBAL_Stream[sno].stream_putc(sno, 256 & (ch >> 8)); GLOBAL_Stream[sno].stream_putc(sno, 256 & ch); return ch; } else { CACHE_REGS Yap_Error(REPRESENTATION_ERROR_CHARACTER, MkIntegerTerm(ch), "charater %ld cannot be encoded in stream %d", (unsigned long int)ch, sno); return -1; } } int put_wchar(int sno, wchar_t ch) { /* pass the bucck if we can */ switch (GLOBAL_Stream[sno].encoding) { case ENC_OCTET: return GLOBAL_Stream[sno].stream_putc(sno, ch); case ENC_ISO_LATIN1: if (ch >= 0xff) { return handle_write_encoding_error(sno, ch); } return GLOBAL_Stream[sno].stream_putc(sno, ch); case ENC_ISO_ASCII: if (ch >= 0x80) { return handle_write_encoding_error(sno, ch); } return GLOBAL_Stream[sno].stream_putc(sno, ch); case ENC_ISO_ANSI: { char buf[MB_LEN_MAX]; mbstate_t mbstate; int n; memset((void *)&mbstate, 0, sizeof(mbstate_t)); if ((n = wcrtomb(buf, ch, &mbstate)) < 0) { /* error */ GLOBAL_Stream[sno].stream_putc(sno, ch); return -1; } else { int i; for (i = 0; i < n; i++) { GLOBAL_Stream[sno].stream_putc(sno, buf[i]); } return ch; } case ENC_ISO_UTF8: if (ch < 0x80) { GLOBAL_Stream[sno].stream_putc(sno, ch); } else if (ch < 0x800) { GLOBAL_Stream[sno].stream_putc(sno, 0xC0 | ch >> 6); GLOBAL_Stream[sno].stream_putc(sno, 0x80 | (ch & 0x3F)); } else if (ch < 0x10000) { GLOBAL_Stream[sno].stream_putc(sno, 0xE0 | ch >> 12); GLOBAL_Stream[sno].stream_putc(sno, 0x80 | (ch >> 6 & 0x3F)); GLOBAL_Stream[sno].stream_putc(sno, 0x80 | (ch & 0x3F)); } else if (ch < 0x200000) { GLOBAL_Stream[sno].stream_putc(sno, 0xF0 | ch >> 18); GLOBAL_Stream[sno].stream_putc(sno, 0x80 | (ch >> 12 & 0x3F)); GLOBAL_Stream[sno].stream_putc(sno, 0x80 | (ch >> 6 & 0x3F)); GLOBAL_Stream[sno].stream_putc(sno, 0x80 | (ch & 0x3F)); } else { /* should never happen */ return -1; } return ch; break; case ENC_UTF16_LE: { if (ch < 0x10000) { GLOBAL_Stream[sno].stream_putc(sno, (ch & 0xff)); GLOBAL_Stream[sno].stream_putc(sno, (ch >> 8)); } else { // computations uint16_t ich = ch; uint16_t lead = LEAD_OFFSET + (ich >> 10); uint16_t trail = 0xDC00 + (ich & 0x3FF); GLOBAL_Stream[sno].stream_putc(sno, (trail & 0xff)); GLOBAL_Stream[sno].stream_putc(sno, (trail >> 8)); GLOBAL_Stream[sno].stream_putc(sno, (lead & 0xff)); GLOBAL_Stream[sno].stream_putc(sno, (lead >> 8)); } return ch; } case ENC_UTF16_BE: { // computations if (ch < 0x10000) { GLOBAL_Stream[sno].stream_putc(sno, (ch >> 8)); GLOBAL_Stream[sno].stream_putc(sno, (ch & 0xff)); } else { uint16_t lead = (uint16_t)LEAD_OFFSET + ((uint16_t)ch >> 10); uint16_t trail = 0xDC00 + ((uint16_t)ch & 0x3FF); GLOBAL_Stream[sno].stream_putc(sno, (lead >> 8)); GLOBAL_Stream[sno].stream_putc(sno, (lead & 0xff)); GLOBAL_Stream[sno].stream_putc(sno, (trail >> 8)); GLOBAL_Stream[sno].stream_putc(sno, (trail & 0xff)); } return ch; } case ENC_UCS2_LE: { if (ch >= 0x10000) { return 0; } GLOBAL_Stream[sno].stream_putc(sno, (ch & 0xff)); GLOBAL_Stream[sno].stream_putc(sno, (ch >> 8)); return ch; } case ENC_UCS2_BE: { // computations if (ch < 0x10000) { GLOBAL_Stream[sno].stream_putc(sno, (ch >> 8)); GLOBAL_Stream[sno].stream_putc(sno, (ch & 0xff)); return ch; } else { return 0; } } case ENC_ISO_UTF32_BE: GLOBAL_Stream[sno].stream_putc(sno, (ch >> 24) & 0xff); GLOBAL_Stream[sno].stream_putc(sno, (ch >> 16) & 0xff); GLOBAL_Stream[sno].stream_putc(sno, (ch >> 8) & 0xff); GLOBAL_Stream[sno].stream_putc(sno, ch & 0xff); return ch; case ENC_ISO_UTF32_LE: GLOBAL_Stream[sno].stream_putc(sno, ch & 0xff); GLOBAL_Stream[sno].stream_putc(sno, (ch >> 8) & 0xff); GLOBAL_Stream[sno].stream_putc(sno, (ch >> 16) & 0xff); GLOBAL_Stream[sno].stream_putc(sno, (ch >> 24) & 0xff); return ch; } } return -1; } /* used by user-code to read characters from the current input stream */ int Yap_PlGetchar(void) { CACHE_REGS return ( GLOBAL_Stream[LOCAL_c_input_stream].stream_getc(LOCAL_c_input_stream)); } int Yap_PlGetWchar(void) { CACHE_REGS return get_wchar(LOCAL_c_input_stream); } /* avoid using a variable to call a function */ int Yap_PlFGetchar(void) { CACHE_REGS return (PlGetc(LOCAL_c_input_stream)); } Term Yap_MkStream(int n) { Term t[1]; t[0] = MkIntTerm(n); return (Yap_MkApplTerm(FunctorStream, 1, t)); } /* given a stream index, get the corresponding fd */ Int GetStreamFd(int sno) { #if HAVE_SOCKET if (GLOBAL_Stream[sno].status & Socket_Stream_f) { return (GLOBAL_Stream[sno].u.socket.fd); } else #endif if (GLOBAL_Stream[sno].status & Pipe_Stream_f) { return (GLOBAL_Stream[sno].u.pipe.fd); } else if (GLOBAL_Stream[sno].status & InMemory_Stream_f) { return (-1); } return (fileno(GLOBAL_Stream[sno].file)); } Int Yap_GetStreamFd(int sno) { return GetStreamFd(sno); } static int binary_file(const char *file_name) { #if HAVE_STAT #if _MSC_VER || defined(__MINGW32__) struct _stat ss; if (_stat(file_name, &ss) != 0) #else struct stat ss; if (stat(file_name, &ss) != 0) #endif { /* ignore errors while checking a file */ return false; } return (S_ISDIR(ss.st_mode)); #else return (FALSE); #endif } static int write_bom(int sno, StreamDesc *st) { /* dump encoding */ switch (st->encoding) { case ENC_ISO_UTF8: if (st->stream_putc(sno, 0xEF) < 0) return false; if (st->stream_putc(sno, 0xBB) < 0) return false; if (st->stream_putc(sno, 0xBF) < 0) return false; st->status |= HAS_BOM_f; return true; case ENC_UTF16_BE: case ENC_UCS2_BE: if (st->stream_putc(sno, 0xFE) < 0) return false; if (st->stream_putc(sno, 0xFF) < 0) return false; st->status |= HAS_BOM_f; return true; case ENC_UTF16_LE: case ENC_UCS2_LE: if (st->stream_putc(sno, 0xFF) < 0) return false; if (st->stream_putc(sno, 0xFE) < 0) return false; st->status |= HAS_BOM_f; return true; case ENC_ISO_UTF32_BE: if (st->stream_putc(sno, 0x00) < 0) return false; if (st->stream_putc(sno, 0x00) < 0) return false; if (st->stream_putc(sno, 0xFE) < 0) return false; if (st->stream_putc(sno, 0xFF) < 0) return false; st->status |= HAS_BOM_f; return true; case ENC_ISO_UTF32_LE: if (st->stream_putc(sno, 0xFF) < 0) return false; if (st->stream_putc(sno, 0xFE) < 0) return false; if (st->stream_putc(sno, 0x00) < 0) return false; if (st->stream_putc(sno, 0x00) < 0) return false; st->status |= HAS_BOM_f; return true; default: return true; } } static void check_bom(int sno, StreamDesc *st) { int ch1, ch2, ch3, ch4; if (st->file == NULL) { PlIOError(SYSTEM_ERROR_INTERNAL, Yap_MkStream(sno), "YAP does not support BOM n %x type of files", st->status); return; } ch1 = st->stream_getc(sno); switch (ch1) { case 0x00: { ch2 = st->stream_getc(sno); if (ch2 != 0x00) { ungetc(ch1, st->file); ungetc(ch2, st->file); return; } else { ch3 = st->stream_getc(sno); if (ch3 == EOFCHAR || ch3 != 0xFE) { ungetc(ch1, st->file); ungetc(ch2, st->file); ungetc(ch3, st->file); return; } else { ch4 = st->stream_getc(sno); if (ch4 == EOFCHAR || ch3 != 0xFF) { ungetc(ch1, st->file); ungetc(ch2, st->file); ungetc(ch3, st->file); ungetc(ch4, st->file); return; } else { st->status |= HAS_BOM_f; st->encoding = ENC_ISO_UTF32_BE; return; } } } } case 0xFE: { ch2 = fgetc(st->file); if (ch2 != 0xFF) { ungetc(ch1, st->file); ungetc(ch2, st->file); return; } else { st->status |= HAS_BOM_f; st->encoding = ENC_UTF16_BE; return; } } case 0xFF: { ch2 = fgetc(st->file); if (ch2 != 0xFE) { ungetc(ch1, st->file); ungetc(ch2, st->file); return; } else { ch3 = fgetc(st->file); if (ch3 != 0x00) { ungetc(ch3, st->file); } else { ch4 = fgetc(st->file); if (ch4 == 0x00) { st->status |= HAS_BOM_f; st->encoding = ENC_ISO_UTF32_LE; return; } else { ungetc(ch4, st->file); ungetc(0x00, st->file); } } } st->status |= HAS_BOM_f; st->encoding = ENC_UTF16_LE; return; } case 0xEF: ch2 = fgetc(st->file); if (ch2 != 0xBB) { ungetc(ch1, st->file); ungetc(ch2, st->file); return; } else { ch3 = fgetc(st->file); if (ch3 != 0xBF) { ungetc(ch1, st->file); ungetc(ch2, st->file); ungetc(ch3, st->file); return; } else { st->status |= HAS_BOM_f; st->encoding = ENC_ISO_UTF8; return; } } default: ungetc(ch1, st->file); } } bool Yap_initStream(int sno, FILE *fd, const char *name, const char *io_mode, Term file_name, encoding_t encoding, stream_flags_t flags, void *vfs) { StreamDesc *st = &GLOBAL_Stream[sno]; __android_log_print(ANDROID_LOG_INFO, "YAPDroid", "init %s %s:%s stream <%d>", io_mode, CurrentModule == 0? "prolog": RepAtom(AtomOfTerm(CurrentModule))->StrOfAE, name, sno); if (io_mode == NULL) Yap_Error(PERMISSION_ERROR_NEW_ALIAS_FOR_STREAM, MkIntegerTerm(sno), "File opened with NULL Permissions"); if (strchr(io_mode, 'a')) { st->status = Append_Stream_f | Output_Stream_f | flags; } else if (strchr(io_mode, 'w')) { st->status = Output_Stream_f | flags; } if (strchr(io_mode, 'r')) { st->status = Input_Stream_f | flags; } if (strchr(io_mode, 'b')) { st->status = Binary_Stream_f | flags; } // st->vfs = vfs; st->buf.on = false; st->charcount = 0; st->linecount = 1; if (flags & Binary_Stream_f) { st->encoding = ENC_OCTET; } else { st->encoding = encoding; } if (name == NULL) { char buf[YAP_FILENAME_MAX + 1]; memset(buf, 0, YAP_FILENAME_MAX + 1); name = Yap_guessFileName(fd, sno, buf, YAP_FILENAME_MAX); } if (!name) Yap_Error(SYSTEM_ERROR_INTERNAL, file_name, "Yap_guessFileName failed: opening a file without a name"); st->name = Yap_LookupAtom(name); st->user_name = file_name; st->file = fd; st->linepos = 0; Yap_DefaultStreamOps(st); return true; } static bool open_header(int sno, Atom open_mode) { if (open_mode == AtomWrite) { const char *ptr; const char s[] = "#!"; int ch; ptr = s; while ((ch = *ptr++)) GLOBAL_Stream[sno].stream_wputc(sno, ch); const char *b = Yap_FindExecutable(); ptr = b; while ((ch = *ptr++)) GLOBAL_Stream[sno].stream_wputc(sno, ch); const char *l = " -L --\n\n YAP script\n#\n# .\n"; ptr = l; while ((ch = *ptr++)) GLOBAL_Stream[sno].stream_wputc(sno, ch); } else if (open_mode == AtomRead) { // skip header int ch; while ((ch = Yap_peek(sno)) == '#') { while ((ch = GLOBAL_Stream[sno].stream_wgetc(sno)) != 10 && ch != -1) ; } } return true; } #define OPEN_DEFS() \ PAR("alias", isatom, OPEN_ALIAS) \ , PAR("bom", booleanFlag, OPEN_BOM), PAR("buffer", isatom, OPEN_BUFFER), \ PAR("close_on_abort", booleanFlag, OPEN_CLOSE_ON_ABORT), \ PAR("create", isatom, OPEN_CREATE), \ PAR("encoding", isatom, OPEN_ENCODING), \ PAR("eof_action", isatom, OPEN_EOF_ACTION), \ PAR("expand_filename", booleanFlag, OPEN_EXPAND_FILENAME), \ PAR("file_name", isatom, OPEN_FILE_NAME), PAR("input", ok, OPEN_INPUT), \ PAR("locale", isatom, OPEN_LOCALE), PAR("lock", isatom, OPEN_LOCK), \ PAR("mode", isatom, OPEN_MODE), PAR("output", ok, OPEN_OUTPUT), \ PAR("representation_errors", booleanFlag, OPEN_REPRESENTATION_ERRORS), \ PAR("reposition", booleanFlag, OPEN_REPOSITION), \ PAR("script", booleanFlag, OPEN_SCRIPT), PAR("type", isatom, OPEN_TYPE), \ PAR("wait", booleanFlag, OPEN_WAIT), PAR(NULL, ok, OPEN_END) #define PAR(x, y, z) z typedef enum open_enum_choices { OPEN_DEFS() } open_choices_t; #undef PAR #define PAR(x, y, z) \ { x, y, z } static const param_t open_defs[] = {OPEN_DEFS()}; #undef PAR static bool fill_stream(int sno, StreamDesc *st, Term tin, const char *io_mode, Term user_name, encoding_t enc) { struct vfs *vfsp = NULL; const char *fname; if (IsAtomTerm(tin)) fname = RepAtom(AtomOfTerm(tin))->StrOfAE; else if (IsStringTerm(tin)) fname = StringOfTerm(tin); else fname = NULL; st->file = NULL; if (fname) { if ((vfsp = vfs_owner(fname)) != NULL && vfsp->open(vfsp, fname, io_mode, sno)) { // read, write, append user_name = st->user_name; st->vfs = vfsp; UNLOCK(st->streamlock); } else { st->file = fopen(fname, io_mode); if (st->file == NULL) { UNLOCK(st->streamlock); if (errno == ENOENT && !strchr(io_mode, 'r')) { PlIOError(EXISTENCE_ERROR_SOURCE_SINK, tin, "%s: %s", fname, strerror(errno)); } else { PlIOError(PERMISSION_ERROR_OPEN_SOURCE_SINK, tin, "%s: %s", fname, strerror(errno)); } } st->vfs = NULL; } if (!st->file && !st->vfs) { PlIOError(EXISTENCE_ERROR_SOURCE_SINK, tin, "%s", fname); /* extract BACK info passed through the stream descriptor */ return false; } } else if (IsApplTerm(tin)) { Functor f = FunctorOfTerm(tin); if (f == FunctorAtom || f == FunctorString || f == FunctorCodes1 || f == FunctorCodes || f == FunctorChars1 || f == FunctorChars) { if (strchr(io_mode, 'r')) { return Yap_OpenBufWriteStream(PASS_REGS1); } else { int i = push_text_stack(); const char *buf; buf = Yap_TextTermToText(tin PASS_REGS); if (!buf) { pop_text_stack(i); return false; } buf = pop_output_text_stack(i, buf); sno = Yap_open_buf_read_stream(buf, strlen(buf) + 1, &LOCAL_encoding, MEM_BUF_MALLOC); return Yap_OpenBufWriteStream(PASS_REGS1); } } else if (!strcmp(RepAtom(NameOfFunctor(f))->StrOfAE, "popen")) { const char *buf; int i = push_text_stack(); buf = Yap_TextTermToText(ArgOfTerm(1, tin) PASS_REGS); if (buf == NULL) { pop_text_stack(i); return false; } #if _WIN32 st->file = _popen(buf, io_mode); #else st->file = popen(buf, io_mode); #endif fname = "popen"; user_name = tin; st->status |= Popen_Stream_f; pop_text_stack(i); } else { Yap_ThrowError(DOMAIN_ERROR_SOURCE_SINK, tin, "open"); } } if (!strchr(io_mode, 'b') && binary_file(fname)) { st->status |= Binary_Stream_f; } Yap_initStream(sno, st->file, fname, io_mode, user_name, LOCAL_encoding, st->status, vfsp); return true; } static Int do_open(Term file_name, Term t2, Term tlist USES_REGS) { Atom open_mode; bool avoid_bom = false, needs_bom = false; Term tenc; char io_mode[8]; int sno = GetFreeStreamD(); if (sno < 0) return (PlIOError(RESOURCE_ERROR_MAX_STREAMS, file_name, "new stream not available for opening")); StreamDesc *st = GLOBAL_Stream + sno; memset(st, 0, sizeof(*st)); // user requested encoding? // BOM mess st->encoding = LOCAL_encoding; if (st->encoding == ENC_UTF16_BE || st->encoding == ENC_UTF16_LE || st->encoding == ENC_UCS2_BE || st->encoding == ENC_UCS2_LE || st->encoding == ENC_ISO_UTF32_BE || st->encoding == ENC_ISO_UTF32_LE) { st->status |= HAS_BOM_f; } st->user_name = Deref(file_name); if (IsVarTerm(file_name)) { Yap_ThrowError(INSTANTIATION_ERROR, file_name, "while opening a list of options"); } // open mode if (IsVarTerm(t2)) { Yap_Error(INSTANTIATION_ERROR, t2, "open/3"); return false; } if (!IsAtomTerm(t2)) { if (IsStringTerm(t2)) { open_mode = Yap_LookupAtom(StringOfTerm(t2)); } else { Yap_Error(TYPE_ERROR_ATOM, t2, "open/3"); return false; } } else { open_mode = AtomOfTerm(t2); } /* get options */ xarg *args = Yap_ArgListToVector(tlist, open_defs, OPEN_END, DOMAIN_ERROR_OPEN_OPTION); if (args == NULL) { if (LOCAL_Error_TYPE != YAP_NO_ERROR) { Yap_Error(LOCAL_Error_TYPE, tlist, "option handling in open/3"); } return false; } /* done */ st->status = 0; const char *s_encoding; if (args[OPEN_ENCODING].used) { tenc = args[OPEN_ENCODING].tvalue; s_encoding = RepAtom(AtomOfTerm(tenc))->StrOfAE; } else { s_encoding = "default"; } // default encoding, no bom yet st->encoding = enc_id(s_encoding, ENC_OCTET); // only set encoding after getting BOM char const *fname0; bool ok = (args[OPEN_EXPAND_FILENAME].used ? args[OPEN_EXPAND_FILENAME].tvalue == TermTrue : false) || trueGlobalPrologFlag(OPEN_EXPANDS_FILENAME_FLAG); if (ok) { if (((IsAtomTerm(file_name) && (fname0 = RepAtom(AtomOfTerm(file_name))->StrOfAE))) || (IsStringTerm(file_name) && (fname0 = StringOfTerm(file_name)))) { int lvl = push_text_stack(); const char *fname = Yap_AbsoluteFile(fname0, ok); file_name = MkAtomTerm(Yap_LookupAtom(fname)); if (!fname) { pop_text_stack(lvl); PlIOError(EXISTENCE_ERROR_SOURCE_SINK, ARG1, NULL); } } } // Skip scripts that start with !#/.. or similar if (open_mode == AtomRead) { strncpy(io_mode, "r", 8); } else if (open_mode == AtomWrite) { strncpy(io_mode, "w", 8); } else if (open_mode == AtomAppend) { strncpy(io_mode, "a", 8); } else { return false; } // binary type if (args[OPEN_TYPE].used) { Term t = args[OPEN_TYPE].tvalue; bool bin = (t == TermBinary); if (bin) { #ifdef _WIN32 strncat(io_mode, "b", 8); #endif st->status |= Binary_Stream_f; st->encoding = ENC_OCTET; avoid_bom = true; needs_bom = false; } else if (t == TermText) { #ifdef _WIN32 strncat(io_mode, "t", 8); #endif /* note that this matters for UNICODE style conversions */ } else { Yap_Error(DOMAIN_ERROR_STREAM_OPTION, t, "type is ~a, must be one of binary or text", t); } } st = &GLOBAL_Stream[sno]; if (!fill_stream(sno, st, file_name,io_mode,st->user_name,st->encoding)) { return false; } if (args[OPEN_BOM].used) { if (args[OPEN_BOM].tvalue == TermTrue) { avoid_bom = false; needs_bom = true; } else if (args[OPEN_BOM].tvalue == TermFalse) { avoid_bom = true; needs_bom = false; } } bool script = (args[OPEN_SCRIPT].used ? args[OPEN_SCRIPT].tvalue == TermTrue : false); if (args[OPEN_ALIAS].used) { Atom al = AtomOfTerm(args[OPEN_ALIAS].tvalue); if (!Yap_AddAlias(al, sno)) { free(args); return false; } } if (st - GLOBAL_Stream < 3) { st->status |= RepError_Prolog_f; } #if MAC if (open_mode == AtomWrite) { Yap_SetTextFile(RepAtom(AtomOfTerm(file_name))->StrOfAE); } #endif // interactive streams do not have a start, so they probably don't have // a BOM avoid_bom = avoid_bom || (st->status & Tty_Stream_f); if (needs_bom && !write_bom(sno, st)) { return false; } else if (open_mode == AtomRead && !avoid_bom) { check_bom(sno, st); // can change encoding // follow declaration unless there is v if (st->status & HAS_BOM_f) { st->encoding = enc_id(s_encoding, st->encoding); } } Yap_DefaultStreamOps(st); if (script) { open_header(sno, open_mode); } free(args); UNLOCK(st->streamlock); { Term t = Yap_MkStream(sno); return (Yap_unify(ARG3, t)); } } /** @pred open(+ _F_,+ _M_,- _S_) is iso Opens the file with name _F_ in mode _M_ (`read`, `write` or `append`), returning _S_ unified with the stream name. Yap allows 64 streams opened at the same time. If you need more, redefine the MaxStreams constant. Each stream is either an input or an output stream but not both. There are always 3 open streams: user_input for reading, user_output for writing and user_error for writing. If there is no ambiguity, the atoms user_input and user_output may be referred to as `user`. The `file_errors` flag controls whether errors are reported when in mode `read` or `append` the file _F_ does not exist or is not readable, and whether in mode `write` or `append` the file is not writable. */ static Int open3(USES_REGS1) { /* '$open'(+File,+Mode,?Stream,-ReturnCode) */ return do_open(Deref(ARG1), Deref(ARG2), TermNil PASS_REGS); } /** @pred open(+ _F_,+ _M_,- _S_,+ _Opts_) is iso Opens the file with name _F_ in mode _M_ (`read`, `write` or `append`), returning _S_ unified with the stream name, and following these options: + `type(+ _T_)` is iso Specify whether the stream is a `text` stream (default), or a `binary` stream. + `reposition(+ _Bool_)` is iso Specify whether it is possible to reposition the stream (`true`), or not (`false`). By default, YAP enables repositioning for all files, except terminal files and sockets. + `eof(+ _Action_)` is iso Specify the action to take if attempting to input characters from a stream where we have previously found an `end_of_file`. The possible actions are `error`, that raises an error, `reset`, that tries to reset the stream and is used for `tty` type files, and `eof_code`, which generates a new `end_of_file` (default for non-tty files). + `alias(+ _Name_)` is iso Specify an alias to the stream. The alias Name must be an atom. The alias can be used instead of the stream descriptor for every operation concerning the stream. The operation will fail and give an error if the alias name is already in use. YAP allows several aliases for the same file, but only one is returned by stream_property/2 + `bom(+ _Bool_)` If present and `true`, a BOM (Byte Order Mark) was detected while opening the file for reading or a BOM was written while opening the stream. See BOM for details. + `encoding(+ _Encoding_)` Set the encoding used for text. See Encoding for an overview of wide character and encoding issues. + `representation_errors(+ _Mode_)` Change the behaviour when writing characters to the stream that cannot be represented by the encoding. The behaviour is one of `error` (throw and Input/Output error exception), `prolog` (write `\u...\` escape code or `xml` (write `\&#...;` XML character entity). The initial mode is `prolog` for the user streams and `error` for all other streams. See also Encoding. + `expand_filename(+ _Mode_)` If _Mode_ is `true` then do filename expansion, then ask Prolog to do file name expansion before actually trying to opening the file: this includes processing `~` characters and processing `$` environment variables at the beginning of the file. Otherwise, just try to open the file using the given name. The default behavior is given by the Prolog flag open_expands_filename. + `script( + _Boolean_ )` YAP extension. The file may be a Prolog script. In `read` mode just check for initial lines if they start with the hash symbol, and skip them. In `write` mode output an header that can be used to launch the file by calling `yap -l file -- $*`. Note that YAP will not set file permissions as executable. In `append` mode ignore the flag. */ static Int open4(USES_REGS1) { /* '$open'(+File,+Mode,?Stream,-ReturnCode) */ return do_open(Deref(ARG1), Deref(ARG2), Deref(ARG4) PASS_REGS); } static Int p_file_expansion(USES_REGS1) { /* '$file_expansion'(+File,-Name) */ Term file_name = Deref(ARG1); /* we know file_name is bound */ if (IsVarTerm(file_name)) { PlIOError(INSTANTIATION_ERROR, file_name, "absolute_file_name/3"); return (FALSE); } int lvl = push_text_stack(); const char *tmp; if ((tmp = Yap_AbsoluteFile(RepAtom(AtomOfTerm(file_name))->StrOfAE, false)) == NULL) { pop_text_stack(lvl); return (PlIOError(EXISTENCE_ERROR_SOURCE_SINK, file_name, "absolute_file_name/3")); } bool rc = (Yap_unify(ARG2, MkAtomTerm(Yap_LookupAtom(tmp)))); pop_text_stack(lvl); return rc; } static Int p_open_null_stream(USES_REGS1) { Term t; StreamDesc *st; int sno = GetFreeStreamD(); if (sno < 0) return (PlIOError(SYSTEM_ERROR_INTERNAL, TermNil, "new stream not available for open_null_stream/1")); st = &GLOBAL_Stream[sno]; st->status = Append_Stream_f | Output_Stream_f | Null_Stream_f; #if _WIN32 st->file = fopen("NUL", "w"); #else st->file = fopen("/dev/null", "w"); #endif if (st->file == NULL) { Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "Could not open NULL stream (/dev/null,NUL)"); return false; } st->linepos = 0; st->charcount = 0; st->linecount = 1; st->stream_putc = NullPutc; st->stream_wputc = put_wchar; st->stream_getc = PlGetc; st->stream_wgetc = get_wchar; st->stream_wgetc_for_read = get_wchar; st->user_name = MkAtomTerm(st->name = AtomDevNull); UNLOCK(st->streamlock); t = Yap_MkStream(sno); return (Yap_unify(ARG1, t)); } int Yap_OpenStream(Term tin, const char *io_mode, Term user_name, encoding_t enc) { CACHE_REGS int sno; StreamDesc *st; sno = GetFreeStreamD(); if (sno < 0) { PlIOError(RESOURCE_ERROR_MAX_STREAMS, tin, "new stream not available for opening"); return -1; } st = GLOBAL_Stream + sno; // fname = Yap_VF(fname); if (fill_stream(sno, st, tin,io_mode,user_name,enc)) return sno; return -1; } int Yap_FileStream(FILE *fd, char *name, Term file_name, int flags, VFS_t *vfsp) { CACHE_REGS int sno; const char *mode; sno = GetFreeStreamD(); if (sno < 0) return (PlIOError(RESOURCE_ERROR_MAX_STREAMS, file_name, "new stream not available for opening")); if (flags & Output_Stream_f) { if (flags & Append_Stream_f) mode = "a"; else mode = "w"; } else { mode = "r"; } Yap_initStream(sno, fd, name, mode, file_name, LOCAL_encoding, flags, vfsp); return sno; } #define CheckStream(arg, kind, msg) \ CheckStream__(__FILE__, __FUNCTION__, __LINE__, arg, kind, msg) static int CheckStream__(const char *file, const char *f, int line, Term arg, int kind, const char *msg) { int sno = -1; arg = Deref(arg); if (IsVarTerm(arg)) { Yap_Error(INSTANTIATION_ERROR, arg, msg); return -1; } else if (IsAtomTerm(arg)) { Atom sname = AtomOfTerm(arg); if (sname == AtomUser) { if (kind & Input_Stream_f) { if (kind & (Output_Stream_f | Append_Stream_f)) { PlIOError__(file, f, line, PERMISSION_ERROR_OUTPUT_STREAM, arg, "ambiguous use of 'user' as a stream"); return (-1); } sname = AtomUserIn; } else { sname = AtomUserOut; } } if ((sno = Yap_CheckAlias(sname)) < 0) { UNLOCK(GLOBAL_Stream[sno].streamlock); PlIOError__(file, f, line, EXISTENCE_ERROR_STREAM, arg, msg); return -1; } else { LOCK(GLOBAL_Stream[sno].streamlock); } } else if (IsApplTerm(arg) && FunctorOfTerm(arg) == FunctorStream) { arg = ArgOfTerm(1, arg); if (!IsVarTerm(arg) && IsIntegerTerm(arg)) { sno = IntegerOfTerm(arg); } } if (sno < 0) { Yap_Error(DOMAIN_ERROR_STREAM_OR_ALIAS, arg, msg); return -1; } if (GLOBAL_Stream[sno].status & Free_Stream_f) { PlIOError__(file, f, line, EXISTENCE_ERROR_STREAM, arg, msg); return -1; } LOCK(GLOBAL_Stream[sno].streamlock); if ((GLOBAL_Stream[sno].status & Input_Stream_f) && !(kind & Input_Stream_f)) { UNLOCK(GLOBAL_Stream[sno].streamlock); PlIOError__(file, f, line, PERMISSION_ERROR_OUTPUT_STREAM, arg, msg); return -1; } if ((GLOBAL_Stream[sno].status & (Append_Stream_f | Output_Stream_f)) && !(kind & Output_Stream_f)) { UNLOCK(GLOBAL_Stream[sno].streamlock); PlIOError__(file, f, line, PERMISSION_ERROR_INPUT_STREAM, arg, msg); return -1; } return sno; } int Yap_CheckStream__(const char *file, const char *f, int line, Term arg, int kind, const char *msg) { return CheckStream__(file, f, line, arg, kind, msg); } int Yap_CheckTextStream__(const char *file, const char *f, int line, Term arg, int kind, const char *msg) { int sno; if ((sno = CheckStream__(file, f, line, arg, kind, msg)) < 0) return -1; if ((GLOBAL_Stream[sno].status & Binary_Stream_f)) { UNLOCK(GLOBAL_Stream[sno].streamlock); if (kind == Input_Stream_f) PlIOError__(file, f, line, PERMISSION_ERROR_INPUT_BINARY_STREAM, arg, msg); else PlIOError__(file, f, line, PERMISSION_ERROR_OUTPUT_BINARY_STREAM, arg, msg); return -1; } return sno; } int Yap_CheckTextWriteStream__(const char *file, const char *f, int line, Term arg, const char *msg) { int sno, kind = Output_Stream_f; if ((sno = CheckStream__(file, f, line, arg, kind, msg)) < 0) return -1; if ((GLOBAL_Stream[sno].status & Binary_Stream_f)) { UNLOCK(GLOBAL_Stream[sno].streamlock); if (kind & Output_Stream_f) PlIOError__(file, f, line, PERMISSION_ERROR_INPUT_BINARY_STREAM, arg, msg); else PlIOError__(file, f, line, PERMISSION_ERROR_OUTPUT_BINARY_STREAM, arg, msg); return -1; } return sno; } int Yap_CheckTextReadStream__(const char *file, const char *f, int line, Term arg, const char *msg) { int sno, kind = Input_Stream_f; if ((sno = CheckStream__(file, f, line, arg, kind, msg)) < 0) return -1; if ((GLOBAL_Stream[sno].status & Binary_Stream_f)) { UNLOCK(GLOBAL_Stream[sno].streamlock); if (kind & Input_Stream_f) PlIOError__(file, f, line, PERMISSION_ERROR_INPUT_BINARY_STREAM, arg, msg); else PlIOError__(file, f, line, PERMISSION_ERROR_OUTPUT_BINARY_STREAM, arg, msg); return -1; } return sno; } int Yap_CheckBinaryStream__(const char *file, const char *f, int line, Term arg, int kind, const char *msg) { int sno; if ((sno = CheckStream__(file, f, line, arg, kind, msg)) < 0) return -1; if (!(GLOBAL_Stream[sno].status & Binary_Stream_f)) { UNLOCK(GLOBAL_Stream[sno].streamlock); if (kind == Input_Stream_f) PlIOError__(file, f, line, PERMISSION_ERROR_INPUT_TEXT_STREAM, arg, msg); else PlIOError__(file, f, line, PERMISSION_ERROR_OUTPUT_TEXT_STREAM, arg, msg); return -1; } return sno; } /* used from C-interface */ int Yap_GetFreeStreamDForReading(void) { int sno = GetFreeStreamD(); StreamDesc *s; if (sno < 0) return sno; s = GLOBAL_Stream + sno; s->status |= User_Stream_f | Input_Stream_f; s->charcount = 0; s->linecount = 1; s->linepos = 0; Yap_DefaultStreamOps(s); UNLOCK(s->streamlock); return sno; } /** * @pred always_prompt_user * * Ensure that the stream always prompts before asking the standard input stream for data. */ static Int always_prompt_user(USES_REGS1) { StreamDesc *s = GLOBAL_Stream + StdInStream; s->status |= Promptable_Stream_f; Yap_DefaultStreamOps(s); return (TRUE); } /** @pred close(+ _S_) is iso Closes the stream _S_. If _S_ does not stand for a stream currently opened an error is reported. The streams user_input, user_output, and user_error can never be closed. */ static Int close1(USES_REGS1) { /* '$close'(+GLOBAL_Stream) */ int sno = CheckStream( ARG1, (Input_Stream_f | Output_Stream_f | Socket_Stream_f), "close/2"); if (sno < 0) return false; if (sno <= StdErrStream) { UNLOCK(GLOBAL_Stream[sno].streamlock); return true; } Yap_CloseStream(sno); UNLOCK(GLOBAL_Stream[sno].streamlock); return (TRUE); } #define CLOSE_DEFS() \ PAR("force", booleanFlag, CLOSE_FORCE), PAR(NULL, ok, CLOSE_END) #define PAR(x, y, z) z typedef enum close_enum_choices { CLOSE_DEFS() } close_choices_t; #undef PAR #define PAR(x, y, z) \ { x, y, z } static const param_t close_defs[] = {CLOSE_DEFS()}; #undef PAR /** @pred close(+ _S_,+ _O_) is iso Closes the stream _S_, following options _O_. The only valid options are `force(true)` and `force(false)`. YAP currently ignores these options. */ static Int close2(USES_REGS1) { /* '$close'(+GLOBAL_Stream) */ Int sno = CheckStream( ARG1, (Input_Stream_f | Output_Stream_f | Socket_Stream_f), "close/2"); Term tlist; if (sno < 0) return (FALSE); if (sno <= StdErrStream) { UNLOCK(GLOBAL_Stream[sno].streamlock); return TRUE; } xarg *args = Yap_ArgListToVector((tlist = Deref(ARG2)), close_defs, CLOSE_END, DOMAIN_ERROR_CLOSE_OPTION); if (args == NULL) { if (LOCAL_Error_TYPE != YAP_NO_ERROR) { Yap_Error(LOCAL_Error_TYPE, tlist, NULL); } return false; } // if (args[CLOSE_FORCE].used) { // } Yap_CloseStream(sno); UNLOCK(GLOBAL_Stream[sno].streamlock); return (TRUE); } Term read_line(int sno) { CACHE_REGS Term tail; Int ch; if ((ch = GLOBAL_Stream[sno].stream_wgetc(sno)) == 10) { return (TermNil); } tail = read_line(sno); return (MkPairTerm(MkIntTerm(ch), tail)); } #define ABSOLUTE_FILE_NAME_DEFS() \ PAR("access", isatom, ABSOLUTE_FILE_NAME_ACCESS) \ , PAR("expand", booleanFlag, ABSOLUTE_FILE_NAME_EXPAND), \ PAR("extensions", ok, ABSOLUTE_FILE_NAME_EXTENSIONS), \ PAR("file_errors", is_file_errors, ABSOLUTE_FILE_NAME_FILE_ERRORS), \ PAR("file_type", is_file_type, ABSOLUTE_FILE_NAME_FILE_TYPE), \ PAR("glob", ok, ABSOLUTE_FILE_NAME_GLOB), \ PAR("relative_to", isatom, ABSOLUTE_FILE_NAME_RELATIVE_TO), \ PAR("solutions", issolutions, ABSOLUTE_FILE_NAME_SOLUTIONS), \ PAR("verbose_file_search", booleanFlag, \ ABSOLUTE_FILE_NAME_VERBOSE_FILE_SEARCH), \ PAR(NULL, ok, ABSOLUTE_FILE_NAME_END) #define PAR(x, y, z) z typedef enum ABSOLUTE_FILE_NAME_enum_ { ABSOLUTE_FILE_NAME_DEFS() } absolute_file_name_choices_t; #undef PAR #define PAR(x, y, z) \ { x, y, z } static const param_t absolute_file_name_search_defs[] = { ABSOLUTE_FILE_NAME_DEFS()}; #undef PAR static Int abs_file_parameters(USES_REGS1) { Term t[ABSOLUTE_FILE_NAME_END]; Term tlist = Deref(ARG1), tf; /* get options */ xarg *args = Yap_ArgListToVector(tlist, absolute_file_name_search_defs, ABSOLUTE_FILE_NAME_END, DOMAIN_ERROR_ABSOLUTE_FILE_NAME_OPTION); if (args == NULL) { if (LOCAL_Error_TYPE != YAP_NO_ERROR) { Yap_Error(LOCAL_Error_TYPE, tlist, NULL); } return false; } /* done */ if (args[ABSOLUTE_FILE_NAME_EXTENSIONS].used) { t[ABSOLUTE_FILE_NAME_EXTENSIONS] = args[ABSOLUTE_FILE_NAME_EXTENSIONS].tvalue; } else { t[ABSOLUTE_FILE_NAME_EXTENSIONS] = TermNil; } if (args[ABSOLUTE_FILE_NAME_RELATIVE_TO].used) { t[ABSOLUTE_FILE_NAME_RELATIVE_TO] = gethdir(args[ABSOLUTE_FILE_NAME_RELATIVE_TO].tvalue); } else { t[ABSOLUTE_FILE_NAME_RELATIVE_TO] = gethdir(TermDot); } if (args[ABSOLUTE_FILE_NAME_FILE_TYPE].used) t[ABSOLUTE_FILE_NAME_FILE_TYPE] = args[ABSOLUTE_FILE_NAME_FILE_TYPE].tvalue; else t[ABSOLUTE_FILE_NAME_FILE_TYPE] = TermTxt; if (args[ABSOLUTE_FILE_NAME_ACCESS].used) t[ABSOLUTE_FILE_NAME_ACCESS] = args[ABSOLUTE_FILE_NAME_ACCESS].tvalue; else t[ABSOLUTE_FILE_NAME_ACCESS] = TermNone; if (args[ABSOLUTE_FILE_NAME_FILE_ERRORS].used) t[ABSOLUTE_FILE_NAME_FILE_ERRORS] = args[ABSOLUTE_FILE_NAME_FILE_ERRORS].tvalue; else t[ABSOLUTE_FILE_NAME_FILE_ERRORS] = TermError; if (args[ABSOLUTE_FILE_NAME_SOLUTIONS].used) t[ABSOLUTE_FILE_NAME_SOLUTIONS] = args[ABSOLUTE_FILE_NAME_SOLUTIONS].tvalue; else t[ABSOLUTE_FILE_NAME_SOLUTIONS] = TermFirst; if (args[ABSOLUTE_FILE_NAME_EXPAND].used) t[ABSOLUTE_FILE_NAME_EXPAND] = args[ABSOLUTE_FILE_NAME_EXPAND].tvalue; else t[ABSOLUTE_FILE_NAME_EXPAND] = TermFalse; if (args[ABSOLUTE_FILE_NAME_GLOB].used) { t[ABSOLUTE_FILE_NAME_GLOB] = args[ABSOLUTE_FILE_NAME_GLOB].tvalue; t[ABSOLUTE_FILE_NAME_EXPAND] = TermTrue; } else t[ABSOLUTE_FILE_NAME_GLOB] = TermEmptyAtom; if (args[ABSOLUTE_FILE_NAME_VERBOSE_FILE_SEARCH].used) t[ABSOLUTE_FILE_NAME_VERBOSE_FILE_SEARCH] = args[ABSOLUTE_FILE_NAME_VERBOSE_FILE_SEARCH].tvalue; else t[ABSOLUTE_FILE_NAME_VERBOSE_FILE_SEARCH] = (trueGlobalPrologFlag(VERBOSE_FILE_SEARCH_FLAG) ? TermTrue : TermFalse); tf = Yap_MkApplTerm(Yap_MkFunctor(AtomOpt, ABSOLUTE_FILE_NAME_END), ABSOLUTE_FILE_NAME_END, t); return (Yap_unify(ARG2, tf)); } static Int get_abs_file_parameter(USES_REGS1) { Term t = Deref(ARG1), topts = Deref(ARG2); /* get options */ /* done */ int i = Yap_ArgKey(AtomOfTerm(t), absolute_file_name_search_defs, ABSOLUTE_FILE_NAME_END); if (i >= 0) return Yap_unify(ARG3, ArgOfTerm(i + 1, topts)); Yap_Error(DOMAIN_ERROR_ABSOLUTE_FILE_NAME_OPTION, ARG1, NULL); return false; } void Yap_InitPlIO(struct yap_boot_params *argi) { Int i; if (argi->inp > 0) Yap_stdin = fdopen(argi->inp - 1, "r"); else if (argi->inp < 0) Yap_stdin = NULL; else Yap_stdin = stdin; if (argi->out > 0) Yap_stdout = fdopen(argi->out - 1, "a"); else if (argi->out < 0) Yap_stdout = NULL; else Yap_stdout = stdout; if (argi->err > 0) Yap_stderr = fdopen(argi->err - 1, "a"); else if (argi->out) Yap_stdout = NULL; else Yap_stderr = stderr; GLOBAL_Stream = (StreamDesc *)Yap_AllocCodeSpace(sizeof(StreamDesc) * MaxStreams); for (i = 0; i < MaxStreams; ++i) { INIT_LOCK(GLOBAL_Stream[i].streamlock); GLOBAL_Stream[i].status = Free_Stream_f; } InitStdStreams(); } void Yap_InitIOPreds(void) { /* here the Input/Output predicates */ Yap_InitCPred("always_prompt_user", 0, always_prompt_user, SafePredFlag | SyncPredFlag); Yap_InitCPred("close", 1, close1, SafePredFlag | SyncPredFlag); Yap_InitCPred("close", 2, close2, SafePredFlag | SyncPredFlag); Yap_InitCPred("open", 4, open4, SyncPredFlag); Yap_InitCPred("open", 3, open3, SyncPredFlag); Yap_InitCPred("abs_file_parameters", 2, abs_file_parameters, SyncPredFlag | HiddenPredFlag); Yap_InitCPred("get_abs_file_parameter", 3, get_abs_file_parameter, SafePredFlag | SyncPredFlag | HiddenPredFlag); Yap_InitCPred("$file_expansion", 2, p_file_expansion, SafePredFlag | SyncPredFlag | HiddenPredFlag); Yap_InitCPred("$open_null_stream", 1, p_open_null_stream, SafePredFlag | SyncPredFlag | HiddenPredFlag); Yap_InitIOStreams(); Yap_InitAliases(); Yap_InitCharsio(); Yap_InitChtypes(); Yap_InitConsole(); Yap_InitReadUtil(); Yap_InitMems(); Yap_InitPipes(); Yap_InitFiles(); Yap_InitWriteTPreds(); Yap_InitReadTPreds(); Yap_InitFormat(); Yap_InitRandomPreds(); #if USE_READLINE Yap_InitReadlinePreds(); #endif Yap_InitSockets(); Yap_InitSignalPreds(); Yap_InitSysPreds(); Yap_InitTimePreds(); }