| 
									
										
										
										
											2016-09-27 19:32:27 -05: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:		mem.c * | 
					
						
							|  |  |  | * Last rev:	5/2/88							 * | 
					
						
							|  |  |  | * mods:									 * | 
					
						
							|  |  |  | * comments:	Input/Output C implemented predicates			 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *************************************************************************/ | 
					
						
							|  |  |  | #ifdef SCCS
 | 
					
						
							|  |  |  | static char SccsId[] = "%W% %G%"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * This file includes the definition of a socket related IO. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "sysbits.h"
 | 
					
						
							|  |  |  | #include "format.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if HAVE_FMEMOPEN
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-28 09:08:22 -05:00
										 |  |  |  int format_synch(int sno, int sno0, format_info *fg) { | 
					
						
							| 
									
										
										
										
											2016-09-27 19:32:27 -05:00
										 |  |  |   const char *s; | 
					
						
							|  |  |  |   int n; | 
					
						
							| 
									
										
										
										
											2016-09-28 09:08:22 -05:00
										 |  |  |   if (sno != sno0) { | 
					
						
							| 
									
										
										
										
											2016-09-27 19:32:27 -05:00
										 |  |  |     fflush(GLOBAL_Stream[sno].file); | 
					
						
							|  |  |  |     n = ftell(GLOBAL_Stream[sno].file); | 
					
						
							| 
									
										
										
										
											2016-09-28 09:08:22 -05:00
										 |  |  |     s = GLOBAL_Stream[sno].nbuf; | 
					
						
							| 
									
										
										
										
											2016-09-27 19:32:27 -05:00
										 |  |  |     fwrite(s, n, 1, GLOBAL_Stream[sno0].file); | 
					
						
							|  |  |  |     rewind(GLOBAL_Stream[sno].file); | 
					
						
							| 
									
										
										
										
											2016-09-28 09:08:22 -05:00
										 |  |  |     fg->lstart = 0; | 
					
						
							|  |  |  |     fg->phys_start = 0; | 
					
						
							|  |  |  |     fg->gapi = 0; | 
					
						
							| 
									
										
										
										
											2016-09-27 19:32:27 -05:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-09-28 09:08:22 -05:00
										 |  |  |   fflush(GLOBAL_Stream[sno0].file); | 
					
						
							|  |  |  |   return sno; | 
					
						
							| 
									
										
										
										
											2016-09-27 19:32:27 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  bool fill_pads(int sno, int sno0, int total, format_info *fg USES_REGS) | 
					
						
							|  |  |  | // uses directly the buffer in the memory stream.
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |   int nfillers, fill_space, lfill_space, nchars; | 
					
						
							|  |  |  |   int (*f_putc)(int, int); | 
					
						
							|  |  |  |   const char *buf; | 
					
						
							|  |  |  |   int phys_end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   f_putc = GLOBAL_Stream[sno0].stream_putc; | 
					
						
							|  |  |  |   if (fflush(GLOBAL_Stream[sno].file) == 0) { | 
					
						
							|  |  |  |     buf = GLOBAL_Stream[sno].nbuf; | 
					
						
							|  |  |  |     phys_end = ftell(GLOBAL_Stream[sno].file); | 
					
						
							|  |  |  |   } else | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   if (fg->gapi == 0) { | 
					
						
							|  |  |  |     fg->gap[0].phys = phys_end; | 
					
						
							|  |  |  |     fg->gap[0].filler = ' '; | 
					
						
							|  |  |  |     fg->gapi = 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   nchars = total - GLOBAL_Stream[sno].linepos; | 
					
						
							|  |  |  |   if (nchars < 0) | 
					
						
							|  |  |  |     nchars = 0; /* ignore */ | 
					
						
							|  |  |  |   nfillers = fg->gapi; | 
					
						
							|  |  |  |   fill_space = nchars / nfillers; | 
					
						
							|  |  |  |   lfill_space = nchars % nfillers; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int i = fg->phys_start; | 
					
						
							|  |  |  |   gap_t *padi = fg->gap; | 
					
						
							|  |  |  |   while (i < phys_end) { | 
					
						
							|  |  |  |     if (i == padi->phys) { | 
					
						
							|  |  |  |       int j; | 
					
						
							|  |  |  |       for (j = 0; j < fill_space; j++) | 
					
						
							|  |  |  |         f_putc(sno0, padi->filler); | 
					
						
							|  |  |  |       padi++; | 
					
						
							|  |  |  |       /* last gap??*/ | 
					
						
							|  |  |  |       if (padi - fg->gap == fg->gapi) { | 
					
						
							|  |  |  |         for (j = 0; j < fill_space; j++) | 
					
						
							|  |  |  |           f_putc(sno0, (padi - 1)->filler); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     f_putc(sno0, buf[i++]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   // final gap
 | 
					
						
							|  |  |  |   if (i == padi->phys) { | 
					
						
							|  |  |  |     int j; | 
					
						
							|  |  |  |     for (j = 0; j < fill_space + lfill_space; j++) | 
					
						
							|  |  |  |       f_putc(sno0, padi->filler); | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   rewind(GLOBAL_Stream[sno].file); | 
					
						
							|  |  |  |   fflush(GLOBAL_Stream[sno0].file); | 
					
						
							|  |  |  |   GLOBAL_Stream[sno].linecount = 1; | 
					
						
							|  |  |  |   GLOBAL_Stream[sno].linepos += nchars; | 
					
						
							|  |  |  |   GLOBAL_Stream[sno].charcount = 0; | 
					
						
							|  |  |  |   fg->phys_start = 0; | 
					
						
							|  |  |  |   fg->lstart = GLOBAL_Stream[sno].linepos; | 
					
						
							|  |  |  |   fg->gapi = 0; | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Yap_set_stream_to_buf(StreamDesc *st, const char *buf, size_t nchars) { | 
					
						
							|  |  |  |   FILE *f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // like any file stream.
 | 
					
						
							|  |  |  |   st->file = f = fmemopen((void *)buf, nchars, "r"); | 
					
						
							| 
									
										
										
										
											2016-11-23 17:23:59 -06:00
										 |  |  |   st->status = Input_Stream_f | InMemory_Stream_f | Seekable_Stream_f; | 
					
						
							| 
									
										
										
										
											2017-06-05 13:06:12 +01:00
										 |  |  |   st->vfs = NULL; | 
					
						
							| 
									
										
										
										
											2016-09-27 19:32:27 -05:00
										 |  |  |   Yap_DefaultStreamOps(st); | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int Yap_open_buf_read_stream(const char *buf, size_t nchars, encoding_t *encp, | 
					
						
							|  |  |  |                              memBufSource src) { | 
					
						
							|  |  |  |   CACHE_REGS | 
					
						
							|  |  |  |   int sno; | 
					
						
							|  |  |  |   StreamDesc *st; | 
					
						
							|  |  |  |   FILE *f; | 
					
						
							|  |  |  |   encoding_t encoding; | 
					
						
							|  |  |  |   stream_flags_t flags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sno = GetFreeStreamD(); | 
					
						
							|  |  |  |   if (sno < 0) | 
					
						
							|  |  |  |     return (PlIOError(RESOURCE_ERROR_MAX_STREAMS, TermNil, | 
					
						
							|  |  |  |                       "new stream not available for open_mem_read_stream/1")); | 
					
						
							|  |  |  |   st = GLOBAL_Stream + sno; | 
					
						
							|  |  |  |   if (encp) | 
					
						
							|  |  |  |     encoding = *encp; | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     encoding = LOCAL_encoding; | 
					
						
							|  |  |  |   // like any file stream.
 | 
					
						
							|  |  |  |   f = st->file = fmemopen((void *)buf, nchars, "r"); | 
					
						
							|  |  |  |   flags = Input_Stream_f | InMemory_Stream_f | Seekable_Stream_f; | 
					
						
							| 
									
										
										
										
											2017-06-05 13:06:12 +01:00
										 |  |  |   Yap_initStream(sno, f, NULL, TermNil, encoding, flags, AtomRead, NULL); | 
					
						
							| 
									
										
										
										
											2016-09-27 19:32:27 -05:00
										 |  |  | // like any file stream.
 | 
					
						
							|  |  |  |   Yap_DefaultStreamOps(st); | 
					
						
							|  |  |  |   UNLOCK(st->streamlock); | 
					
						
							|  |  |  |   return sno; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int | 
					
						
							|  |  |  | open_mem_read_stream(USES_REGS1) /* $open_mem_read_stream(+List,-Stream) */ | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Term t, ti; | 
					
						
							|  |  |  |   int sno; | 
					
						
							|  |  |  |   char buf0[YAP_FILENAME_MAX + 1]; | 
					
						
							|  |  |  |   const char *buf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ti = Deref(ARG1); | 
					
						
							|  |  |  |   buf = Yap_TextTermToText(ti, buf0, 0, LOCAL_encoding); | 
					
						
							|  |  |  |   if (!buf) { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   sno = Yap_open_buf_read_stream(buf, strlen(buf) + 1, &LOCAL_encoding, | 
					
						
							|  |  |  |                                  MEM_BUF_MALLOC); | 
					
						
							|  |  |  |   t = Yap_MkStream(sno); | 
					
						
							|  |  |  |   return Yap_unify(ARG2, t); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // open a buffer for writing, currently just ignores buf and nchars.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int Yap_open_buf_write_stream(encoding_t enc, memBufSource src) { | 
					
						
							|  |  |  |   CACHE_REGS | 
					
						
							|  |  |  |   int sno; | 
					
						
							|  |  |  |   StreamDesc *st; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sno = GetFreeStreamD(); | 
					
						
							|  |  |  |   if (sno < 0) | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  |   st = GLOBAL_Stream + sno; | 
					
						
							|  |  |  |   st->status = Output_Stream_f | InMemory_Stream_f | FreeOnClose_Stream_f; | 
					
						
							|  |  |  |   st->linepos = 0; | 
					
						
							|  |  |  |   st->charcount = 0; | 
					
						
							|  |  |  |   st->linecount = 1; | 
					
						
							|  |  |  |   st->encoding = enc; | 
					
						
							| 
									
										
										
										
											2017-06-05 13:06:12 +01:00
										 |  |  |   st->vfs = NULL; | 
					
						
							| 
									
										
										
										
											2016-09-27 19:32:27 -05:00
										 |  |  |   Yap_DefaultStreamOps(st); | 
					
						
							|  |  |  | #if HAVE_OPEN_MEMSTREAM
 | 
					
						
							|  |  |  |   st->file = open_memstream(&st->nbuf, &st->nsize); | 
					
						
							|  |  |  |   // setbuf(st->file, NULL);
 | 
					
						
							|  |  |  |   st->status |= Seekable_Stream_f; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   st->file = fmemopen((void *)buf, nchars, "w"); | 
					
						
							|  |  |  |   st->nsize = nchars; | 
					
						
							|  |  |  |   st->nbuf = buf; | 
					
						
							|  |  |  |   if (!st->nbuf) { | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   UNLOCK(st->streamlock); | 
					
						
							|  |  |  |   return sno; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int Yap_OpenBufWriteStream(USES_REGS1) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return Yap_open_buf_write_stream( | 
					
						
							|  |  |  |       GLOBAL_Stream[LOCAL_c_output_stream].encoding, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int | 
					
						
							|  |  |  | open_mem_write_stream(USES_REGS1) /* $open_mem_write_stream(-Stream) */ | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Term t; | 
					
						
							|  |  |  |   int sno; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sno = Yap_OpenBufWriteStream(PASS_REGS1); | 
					
						
							|  |  |  |   if (sno == -1) | 
					
						
							|  |  |  |     return (PlIOError(SYSTEM_ERROR_INTERNAL, TermNil, | 
					
						
							|  |  |  |                       "new stream not available for open_mem_read_stream/1")); | 
					
						
							|  |  |  |   t = Yap_MkStream(sno); | 
					
						
							|  |  |  |   GLOBAL_Stream[sno].status |= InMemory_Stream_f; | 
					
						
							|  |  |  |   return (Yap_unify(ARG1, t)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Yap_PeekMemwriteStream() shows the current buffer for a memory stream. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param sno, the in-memory stream | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @return temporary buffer, discarded by close and may be moved away | 
					
						
							|  |  |  |  * by other writes.. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | char *Yap_MemExportStreamPtr(int sno) { | 
					
						
							|  |  |  |   char *s; | 
					
						
							|  |  |  |   if (fflush(GLOBAL_Stream[sno].file) == 0) { | 
					
						
							|  |  |  |     s = GLOBAL_Stream[sno].nbuf; | 
					
						
							|  |  |  |     // s[fseek(GLOBAL_Stream[sno].file, 0, SEEK_END)] = '\0';
 | 
					
						
							|  |  |  |     return s; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int peek_mem_write_stream( | 
					
						
							|  |  |  |     USES_REGS1) { /* '$peek_mem_write_stream'(+GLOBAL_Stream,?S0,?S) */ | 
					
						
							|  |  |  |   Int sno = | 
					
						
							|  |  |  |       Yap_CheckStream(ARG1, (Output_Stream_f | InMemory_Stream_f), "close/2"); | 
					
						
							|  |  |  |   Int i; | 
					
						
							|  |  |  |   Term tf = ARG2; | 
					
						
							|  |  |  |   CELL *HI; | 
					
						
							|  |  |  |   const char *ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (sno < 0) | 
					
						
							|  |  |  |     return (FALSE); | 
					
						
							|  |  |  | restart: | 
					
						
							|  |  |  |   HI = HR; | 
					
						
							|  |  |  |   if (fflush(GLOBAL_Stream[sno].file) == 0) { | 
					
						
							|  |  |  |     i = fseek(GLOBAL_Stream[sno].file, 0, SEEK_END); | 
					
						
							|  |  |  |     ptr = GLOBAL_Stream[sno].nbuf; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   while (i > 0) { | 
					
						
							|  |  |  |     --i; | 
					
						
							|  |  |  |     tf = MkPairTerm(MkIntTerm(ptr[i]), tf); | 
					
						
							|  |  |  |     if (HR + 1024 >= ASP) { | 
					
						
							|  |  |  |       UNLOCK(GLOBAL_Stream[sno].streamlock); | 
					
						
							|  |  |  |       HR = HI; | 
					
						
							|  |  |  |       if (!Yap_gcl((ASP - HI) * sizeof(CELL), 3, ENV, Yap_gcP())) { | 
					
						
							|  |  |  |         UNLOCK(GLOBAL_Stream[sno].streamlock); | 
					
						
							|  |  |  |         Yap_Error(RESOURCE_ERROR_STACK, TermNil, LOCAL_ErrorMessage); | 
					
						
							|  |  |  |         return (FALSE); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       i = GLOBAL_Stream[sno].u.mem_string.pos; | 
					
						
							|  |  |  |       tf = ARG2; | 
					
						
							|  |  |  |       LOCK(GLOBAL_Stream[sno].streamlock); | 
					
						
							|  |  |  |       goto restart; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   UNLOCK(GLOBAL_Stream[sno].streamlock); | 
					
						
							|  |  |  |   return (Yap_unify(ARG3, tf)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Yap_MemOps(StreamDesc *st) { | 
					
						
							|  |  |  |   st->stream_putc = FilePutc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   st->stream_getc = PlGetc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Yap_CloseMemoryStream(int sno) { | 
					
						
							|  |  |  |   if ((GLOBAL_Stream[sno].status & Output_Stream_f)) { | 
					
						
							|  |  |  |     fflush(GLOBAL_Stream[sno].file); | 
					
						
							|  |  |  |     fclose(GLOBAL_Stream[sno].file); | 
					
						
							|  |  |  |     if (GLOBAL_Stream[sno].status & FreeOnClose_Stream_f) | 
					
						
							|  |  |  |       free(GLOBAL_Stream[sno].nbuf); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     fclose(GLOBAL_Stream[sno].file); | 
					
						
							|  |  |  |     if (GLOBAL_Stream[sno].status & FreeOnClose_Stream_f) | 
					
						
							|  |  |  |       free(GLOBAL_Stream[sno].nbuf); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Yap_InitMems(void) { | 
					
						
							|  |  |  |   CACHE_REGS | 
					
						
							|  |  |  |   Yap_InitCPred("open_mem_read_stream", 2, open_mem_read_stream, SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("open_mem_write_stream", 1, open_mem_write_stream, | 
					
						
							|  |  |  |                 SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("peek_mem_write_stream", 3, peek_mem_write_stream, | 
					
						
							|  |  |  |                 SyncPredFlag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 |