| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | /************************************************************************* | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |  *									 * | 
					
						
							|  |  |  |  *	 YAP Prolog 							 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  *	Yap Prolog was developed at NCCUP - Universidade do Porto	 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  * Copyright L.Damas, V. Santos Costa and Universidade do Porto 1985--	 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  ************************************************************************** | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  * File:		strings.c * | 
					
						
							|  |  |  |  * comments:	General-conversion of character sequences.		 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  * Last rev:     $Date: 2008-07-24 16:02:00 $,$Author: vsc $	     	 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  *************************************************************************/ | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "Yap.h" | 
					
						
							|  |  |  | #include "YapEval.h" | 
					
						
							|  |  |  | #include "YapHeap.h" | 
					
						
							|  |  |  | #include "YapText.h" | 
					
						
							|  |  |  | #include "Yatom.h" | 
					
						
							|  |  |  | #include "yapio.h" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <YapText.h> | 
					
						
							|  |  |  | #include <string.h> | 
					
						
							|  |  |  | #include <wchar.h> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef HAVE_WCSNLEN | 
					
						
							|  |  |  | inline static size_t min_size(size_t i, size_t j) { return (i < j ? i : j); } | 
					
						
							|  |  |  | #define wcsnlen(S, N) min_size(N, wcslen(S)) | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | #if !defined(HAVE_STPCPY) && !defined(__APPLE__) | 
					
						
							|  |  |  | inline static void *__stpcpy(void *i, const void *j) { | 
					
						
							|  |  |  |   return strcpy(i, j) + strlen(j); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #define stpcpy __stpcpy | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | #ifndef NAN | 
					
						
							|  |  |  | #define NAN (0.0 / 0.0) | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define MAX_PATHNAME 2048 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct mblock { | 
					
						
							|  |  |  |   struct mblock *prev, *next; | 
					
						
							|  |  |  |   int lvl; | 
					
						
							|  |  |  |   size_t sz; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct TextBuffer_manager { | 
					
						
							|  |  |  |   void *buf, *ptr; | 
					
						
							|  |  |  |   size_t sz; | 
					
						
							|  |  |  |   struct mblock *first[16]; | 
					
						
							|  |  |  |   struct mblock *last[16]; | 
					
						
							|  |  |  |   int lvl; | 
					
						
							|  |  |  | } text_buffer_t; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | int AllocLevel(void) { return LOCAL_TextBuffer->lvl; } | 
					
						
							|  |  |  | int push_text_stack__(USES_REGS1) { | 
					
						
							|  |  |  |   int i = LOCAL_TextBuffer->lvl; | 
					
						
							|  |  |  |   i++; | 
					
						
							|  |  |  |   LOCAL_TextBuffer->lvl = i; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   return i; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | int pop_text_stack__(int i) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   int lvl = LOCAL_TextBuffer->lvl; | 
					
						
							|  |  |  |   while (lvl >= i) { | 
					
						
							|  |  |  |     struct mblock *p = LOCAL_TextBuffer->first[lvl]; | 
					
						
							|  |  |  |     while (p) { | 
					
						
							|  |  |  |       struct mblock *np = p->next; | 
					
						
							|  |  |  |       free(p); | 
					
						
							|  |  |  |       p = np; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     LOCAL_TextBuffer->first[lvl] = NULL; | 
					
						
							|  |  |  |     LOCAL_TextBuffer->last[lvl] = NULL; | 
					
						
							|  |  |  |     lvl--; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   LOCAL_TextBuffer->lvl = lvl; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   return lvl; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | void *pop_output_text_stack__(int i, const void *export) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   int lvl = LOCAL_TextBuffer->lvl; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   while (lvl >= i) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     struct mblock *p = LOCAL_TextBuffer->first[lvl]; | 
					
						
							|  |  |  |     while (p) { | 
					
						
							|  |  |  |       struct mblock *np = p->next; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       if (p + 1 == export) { | 
					
						
							|  |  |  | 	size_t sz = p->sz - sizeof(struct mblock); | 
					
						
							|  |  |  | 	memmove(p, p + 1, sz); | 
					
						
							|  |  |  | 	export = p; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 	free(p); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |       p = np; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     LOCAL_TextBuffer->first[lvl] = NULL; | 
					
						
							|  |  |  |     LOCAL_TextBuffer->last[lvl] = NULL; | 
					
						
							|  |  |  |     lvl--; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   LOCAL_TextBuffer->lvl = lvl; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   return (void *)export; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //	void pop_text_stack(int i) { LOCAL_TextBuffer->lvl = i; } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | void insert_block(struct mblock *o) { | 
					
						
							|  |  |  |   int lvl = o->lvl; | 
					
						
							|  |  |  |   o->prev = LOCAL_TextBuffer->last[lvl]; | 
					
						
							|  |  |  |   if (o->prev) { | 
					
						
							|  |  |  |     o->prev->next = o; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (LOCAL_TextBuffer->first[lvl]) { | 
					
						
							|  |  |  |     LOCAL_TextBuffer->last[lvl] = o; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     LOCAL_TextBuffer->first[lvl] = LOCAL_TextBuffer->last[lvl] = o; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   o->next = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void release_block(struct mblock *o) { | 
					
						
							|  |  |  |   int lvl = o->lvl; | 
					
						
							|  |  |  |   if (LOCAL_TextBuffer->first[lvl] == o) { | 
					
						
							|  |  |  |     if (LOCAL_TextBuffer->last[lvl] == o) { | 
					
						
							|  |  |  |       LOCAL_TextBuffer->first[lvl] = LOCAL_TextBuffer->last[lvl] = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     LOCAL_TextBuffer->first[lvl] = o->next; | 
					
						
							|  |  |  |   } else if (LOCAL_TextBuffer->last[lvl] == o) { | 
					
						
							|  |  |  |     LOCAL_TextBuffer->last[lvl] = o->prev; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (o->prev) | 
					
						
							|  |  |  |     o->prev->next = o->next; | 
					
						
							|  |  |  |   if (o->next) | 
					
						
							|  |  |  |     o->next->prev = o->prev; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | void *Malloc(size_t sz USES_REGS) { | 
					
						
							|  |  |  |   int lvl = LOCAL_TextBuffer->lvl; | 
					
						
							|  |  |  |   if (sz == 0) | 
					
						
							|  |  |  |     sz = 1024; | 
					
						
							|  |  |  |   sz = ALIGN_BY_TYPE(sz + sizeof(struct mblock), CELL); | 
					
						
							|  |  |  |   struct mblock *o = malloc(sz); | 
					
						
							|  |  |  |   if (!o) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   o->sz = sz; | 
					
						
							|  |  |  |   o->lvl = lvl; | 
					
						
							|  |  |  |   o->prev = o->next = 0; | 
					
						
							|  |  |  |   insert_block(o); | 
					
						
							|  |  |  |   return o + 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void *MallocAtLevel(size_t sz, int atL USES_REGS) { | 
					
						
							|  |  |  |   int lvl = LOCAL_TextBuffer->lvl; | 
					
						
							|  |  |  |   if (atL > 0 && atL <= lvl) { | 
					
						
							|  |  |  |     lvl = atL; | 
					
						
							|  |  |  |   } else if (atL < 0 && lvl - atL >= 0) { | 
					
						
							|  |  |  |     lvl += atL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (sz == 0) | 
					
						
							|  |  |  |     sz = 1024; | 
					
						
							|  |  |  |   sz = ALIGN_BY_TYPE(sz + sizeof(struct mblock), CELL); | 
					
						
							|  |  |  |   struct mblock *o = malloc(sz); | 
					
						
							|  |  |  |   if (!o) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   o->sz = sz; | 
					
						
							|  |  |  |   o->lvl = lvl; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   o->prev = o->next = 0; | 
					
						
							|  |  |  |   insert_block(o); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   return o + 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void *Realloc(void *pt, size_t sz USES_REGS) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   sz += sizeof(struct mblock); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   struct mblock *old = pt, *o; | 
					
						
							|  |  |  |   old--; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   release_block(old); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   o = realloc(old, sz); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   o->sz = sz; | 
					
						
							|  |  |  |   insert_block(o); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return o + 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Export a local memory object as a RO object to the outside world, that is, | 
					
						
							|  |  |  |  * recovering as much storage as one can. | 
					
						
							|  |  |  |  * @param pt pointer to object | 
					
						
							|  |  |  |  * @return new object | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const void *MallocExportAsRO(const void *pt USES_REGS) { | 
					
						
							|  |  |  |   struct mblock *old = (void *)pt, *o = old - 1; | 
					
						
							|  |  |  |   if (old == NULL) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   size_t sz = o->sz; | 
					
						
							|  |  |  |   release_block(o); | 
					
						
							|  |  |  |   memmove((void *)o, pt, sz); | 
					
						
							|  |  |  |   return realloc((void *)o, sz); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | void Free(void *pt USES_REGS) { | 
					
						
							|  |  |  |   struct mblock *o = pt; | 
					
						
							|  |  |  |   o--; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   release_block(o); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   free(o); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void *Yap_InitTextAllocator(void) { | 
					
						
							|  |  |  |   struct TextBuffer_manager *new = calloc(sizeof(struct TextBuffer_manager), 1); | 
					
						
							|  |  |  |   return new; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static size_t MaxTmp(USES_REGS1) { return 1025; } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static Term Globalize(Term v USES_REGS) { | 
					
						
							|  |  |  |   if (!IsVarTerm(v = Deref(v))) { | 
					
						
							|  |  |  |     return v; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (VarOfTerm(v) > HR && VarOfTerm(v) < LCL0) { | 
					
						
							|  |  |  |     Bind_Local(VarOfTerm(v), MkVarTerm()); | 
					
						
							|  |  |  |     v = Deref(v); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return v; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static void *codes2buf(Term t0, void *b0, bool get_codes, | 
					
						
							|  |  |  | 		       bool fixed USES_REGS) { | 
					
						
							|  |  |  |   unsigned char *st0, *st, ar[16]; | 
					
						
							|  |  |  |   Term t = Deref(t0); | 
					
						
							|  |  |  |   size_t length = 0; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (t == TermNil) { | 
					
						
							|  |  |  |     st0 = Malloc(4); | 
					
						
							|  |  |  |     st0[0] = 0; | 
					
						
							|  |  |  |     return st0; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (!IsPairTerm(t)) { | 
					
						
							|  |  |  |     Yap_ThrowError(TYPE_ERROR_LIST, t, "scanning list of codes"); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   bool codes = IsIntegerTerm(HeadOfTerm(t)); | 
					
						
							|  |  |  |   if (get_codes != codes && fixed) { | 
					
						
							|  |  |  |     if (codes) { | 
					
						
							|  |  |  |       Yap_ThrowError(TYPE_ERROR_INTEGER, HeadOfTerm(t), | 
					
						
							|  |  |  | 		     "scanning list of codes"); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       Yap_ThrowError(TYPE_ERROR_ATOM, HeadOfTerm(t), "scanning list of atoms"); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (codes) { | 
					
						
							|  |  |  |     while (IsPairTerm(t)) { | 
					
						
							|  |  |  |       Term hd = HeadOfTerm(t); | 
					
						
							|  |  |  |       if (IsVarTerm(hd)) { | 
					
						
							|  |  |  | 	Yap_ThrowError(INSTANTIATION_ERROR, hd, "scanning list of codes"); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       if (!IsIntegerTerm(hd)) { | 
					
						
							|  |  |  | 	Yap_ThrowError(TYPE_ERROR_CHARACTER_CODE, hd, "scanning list of codes"); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       Int code = IntegerOfTerm(hd); | 
					
						
							|  |  |  |       if (code < 0) { | 
					
						
							|  |  |  | 	Yap_ThrowError(REPRESENTATION_ERROR_CHARACTER_CODE, hd, | 
					
						
							|  |  |  | 		       "scanning list of character codes, found %d", code); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       length += put_utf8(ar, code); | 
					
						
							|  |  |  |       t = TailOfTerm(t); | 
					
						
							|  |  |  |       if (IsVarTerm(t)) { | 
					
						
							|  |  |  | 	Yap_ThrowError(INSTANTIATION_ERROR, t, "scanning list of codes"); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (!IsPairTerm(t) && t != TermNil) { | 
					
						
							|  |  |  | 	Yap_ThrowError(TYPE_ERROR_LIST, t, "scanning list of codes"); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     while (IsPairTerm(t)) { | 
					
						
							|  |  |  |       Term hd = HeadOfTerm(t); | 
					
						
							|  |  |  |       if (IsVarTerm(hd)) { | 
					
						
							|  |  |  | 	Yap_ThrowError(INSTANTIATION_ERROR, hd, "scanning list of codes"); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (!IsAtomTerm(hd)) { | 
					
						
							|  |  |  | 	Yap_ThrowError(TYPE_ERROR_CHARACTER, hd, "scanning list of texts"); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       const char *code = RepAtom(AtomOfTerm(hd))->StrOfAE; | 
					
						
							|  |  |  |       if (code < 0) { | 
					
						
							|  |  |  | 	Yap_ThrowError(TYPE_ERROR_CHARACTER, hd, "scanning list of atoms"); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       length += strlen(code); | 
					
						
							|  |  |  |       t = TailOfTerm(t); | 
					
						
							|  |  |  |       if (IsVarTerm(t)) { | 
					
						
							|  |  |  | 	Yap_ThrowError(INSTANTIATION_ERROR, t, "scanning list of codes"); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       if (!IsPairTerm(t) && t != TermNil) { | 
					
						
							|  |  |  | 	Yap_ThrowError(TYPE_ERROR_LIST, t, "scanning list of codes"); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!IsVarTerm(t)) { | 
					
						
							|  |  |  |     if (t != TermNil) { | 
					
						
							|  |  |  |       Yap_ThrowError(TYPE_ERROR_LIST, t, "scanning list of codes"); | 
					
						
							|  |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   st0 = st = Malloc(length + 1); | 
					
						
							|  |  |  |   t = t0; | 
					
						
							|  |  |  |   if (codes) { | 
					
						
							|  |  |  |     while (IsPairTerm(t)) { | 
					
						
							|  |  |  |       Term hd = HeadOfTerm(t); | 
					
						
							|  |  |  |       Int code = IntegerOfTerm(hd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       st = st + put_utf8(st, code); | 
					
						
							|  |  |  |       t = TailOfTerm(t); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     while (IsPairTerm(t)) { | 
					
						
							|  |  |  |       Term hd = HeadOfTerm(t); | 
					
						
							|  |  |  |       const char *code = RepAtom(AtomOfTerm(hd))->StrOfAE; | 
					
						
							|  |  |  |       st = (unsigned char *)stpcpy((char *)st, code); | 
					
						
							|  |  |  |       t = TailOfTerm(t); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   st[0] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   return st0; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static unsigned char *latin2utf8(seq_tv_t *inp) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   unsigned char *b0 = inp->val.uc; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   size_t sz = strlen(inp->val.c); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   sz *= 2; | 
					
						
							|  |  |  |   int ch; | 
					
						
							|  |  |  |   unsigned char *buf = Malloc(sz + 1), *pt = buf; | 
					
						
							|  |  |  |   if (!buf) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   while ((ch = *b0++)) { | 
					
						
							|  |  |  |     int off = put_utf8(pt, ch); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     if (off < 0) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       continue; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     pt += off; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   *pt++ = '\0'; | 
					
						
							|  |  |  |   return buf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static unsigned char *wchar2utf8(seq_tv_t *inp) { | 
					
						
							|  |  |  |   size_t sz = wcslen(inp->val.w) * 4; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   wchar_t *b0 = inp->val.w; | 
					
						
							|  |  |  |   unsigned char *buf = Malloc(sz + 1), *pt = buf; | 
					
						
							|  |  |  |   int ch; | 
					
						
							|  |  |  |   if (!buf) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   while ((ch = *b0++)) | 
					
						
							|  |  |  |     pt += put_utf8(pt, ch); | 
					
						
							|  |  |  |   *pt++ = '\0'; | 
					
						
							|  |  |  |   return buf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static void *slice(size_t min, size_t max, const unsigned char *buf USES_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static unsigned char *Yap_ListOfCodesToBuffer(unsigned char *buf, Term t, | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 					      seq_tv_t *inp USES_REGS) { | 
					
						
							|  |  |  |   bool codes = true, fixed = true; | 
					
						
							|  |  |  |   unsigned char *nbuf = codes2buf(t, buf, codes, fixed PASS_REGS); | 
					
						
							|  |  |  |   return nbuf; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static unsigned char *Yap_ListOfAtomsToBuffer(unsigned char *buf, Term t, | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 					      seq_tv_t *inp USES_REGS) { | 
					
						
							|  |  |  |   bool codes = false; | 
					
						
							|  |  |  |   unsigned char *nbuf = codes2buf(t, buf, codes, true PASS_REGS); | 
					
						
							|  |  |  |   return nbuf; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static unsigned char *Yap_ListToBuffer(unsigned char *buf, Term t, | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 				       seq_tv_t *inp USES_REGS) { | 
					
						
							|  |  |  |   return codes2buf(t, buf, NULL, false PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if USE_GEN_TYPE_ERROR | 
					
						
							|  |  |  | static yap_error_number gen_type_error(int flags) { | 
					
						
							|  |  |  |   if ((flags & (YAP_STRING_STRING | YAP_STRING_ATOM | YAP_STRING_INT | | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 		YAP_STRING_FLOAT | YAP_STRING_ATOMS_CODES | YAP_STRING_BIG)) == | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       (YAP_STRING_STRING | YAP_STRING_ATOM | YAP_STRING_INT | YAP_STRING_FLOAT | | 
					
						
							|  |  |  |        YAP_STRING_ATOMS_CODES | YAP_STRING_BIG)) | 
					
						
							|  |  |  |     return TYPE_ERROR_TEXT; | 
					
						
							|  |  |  |   if ((flags & (YAP_STRING_STRING | YAP_STRING_ATOM | YAP_STRING_INT | | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 		YAP_STRING_FLOAT | YAP_STRING_BIG)) == | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       (YAP_STRING_STRING | YAP_STRING_ATOM | YAP_STRING_INT | YAP_STRING_FLOAT | | 
					
						
							|  |  |  |        YAP_STRING_BIG)) | 
					
						
							|  |  |  |     return TYPE_ERROR_ATOMIC; | 
					
						
							|  |  |  |   if ((flags & (YAP_STRING_INT | YAP_STRING_FLOAT | YAP_STRING_BIG)) == | 
					
						
							|  |  |  |       (YAP_STRING_INT | YAP_STRING_FLOAT | YAP_STRING_BIG)) | 
					
						
							|  |  |  |     return TYPE_ERROR_NUMBER; | 
					
						
							|  |  |  |   if (flags & YAP_STRING_ATOM) | 
					
						
							|  |  |  |     return TYPE_ERROR_ATOM; | 
					
						
							|  |  |  |   if (flags & YAP_STRING_STRING) | 
					
						
							|  |  |  |     return TYPE_ERROR_STRING; | 
					
						
							|  |  |  |   if (flags & (YAP_STRING_CODES | YAP_STRING_ATOMS)) | 
					
						
							|  |  |  |     return TYPE_ERROR_LIST; | 
					
						
							|  |  |  |   return TYPE_ERROR_NUMBER; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static  yap_error_number type_error(seq_tv_t *inp) { | 
					
						
							|  |  |  |   if ((inp->type & YAP_STRING_ATOMS_CODES) == YAP_STRING_ATOMS_CODES) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       return TYPE_ERROR_IN_CHARACTER; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if ((inp->type & YAP_STRING_ATOMS) == YAP_STRING_ATOMS) { | 
					
						
							|  |  |  |     return TYPE_ERROR_CHARACTER; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if ((inp->type & YAP_STRING_CODES) == YAP_STRING_CODES) { | 
					
						
							|  |  |  |     return TYPE_ERROR_CHARACTER_CODE; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if ((inp->type & YAP_STRING_STRING) == YAP_STRING_STRING) { | 
					
						
							|  |  |  |     return TYPE_ERROR_IN_CHARACTER; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if ((inp->type & YAP_STRING_CODES) == YAP_STRING_CODES) { | 
					
						
							|  |  |  |     return TYPE_ERROR_IN_CHARACTER; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if ((inp->type & (YAP_STRING_ATOM|YAP_STRING_STRING)) ==  (YAP_STRING_ATOM|YAP_STRING_STRING)) { | 
					
						
							|  |  |  |     return TYPE_ERROR_ATOMIC; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if ((inp->type & YAP_STRING_ATOM) == YAP_STRING_ATOM) { | 
					
						
							|  |  |  |     return TYPE_ERROR_IN_CHARACTER; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   return SYSTEM_ERROR_INTERNAL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// @brief translate anything to bufffer UTF-8 | 
					
						
							|  |  |  | /// | 
					
						
							|  |  |  | /// @arg input descriptor | 
					
						
							|  |  |  | unsigned char *Yap_readText(seq_tv_t *inp USES_REGS) { | 
					
						
							|  |  |  | #define POPRET(x) return pop_output_text_stack(lvl, x) | 
					
						
							|  |  |  |   int lvl = push_text_stack(); | 
					
						
							|  |  |  |   char *out = NULL; | 
					
						
							|  |  |  |   /* we know what the term is */ | 
					
						
							|  |  |  |   if ((inp->type & (YAP_STRING_CHARS | YAP_STRING_WCHARS))) { | 
					
						
							|  |  |  |     //> buffer processing | 
					
						
							|  |  |  |     //> must convert: | 
					
						
							|  |  |  |     if (inp->type & YAP_STRING_CHARS) { | 
					
						
							|  |  |  |       //> - ASCII to UTF-8 | 
					
						
							|  |  |  |       if (inp->enc == ENC_ISO_ASCII) { | 
					
						
							|  |  |  | 	pop_text_stack(lvl); | 
					
						
							|  |  |  | 	return inp->val.uc; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       //> - ISO-LATIN-1 to UTF-8 | 
					
						
							|  |  |  |       if (inp->enc == ENC_ISO_LATIN1) { | 
					
						
							|  |  |  | 	POPRET( (char*)latin2utf8(inp)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       //> - assume UTF-8 | 
					
						
							|  |  |  |       if (inp->enc == ENC_ISO_UTF8) { | 
					
						
							|  |  |  | 	pop_text_stack(lvl); | 
					
						
							|  |  |  | 	return inp->val.uc; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     if (inp->type & YAP_STRING_WCHARS) { | 
					
						
							|  |  |  |       // printf("%S\n",inp->val.w); | 
					
						
							|  |  |  |       //> wide includes everything else | 
					
						
							|  |  |  |       POPRET( (char *)wchar2utf8(inp) ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (inp->type & YAP_STRING_TERM) { | 
					
						
							|  |  |  |       //> term: anything goes | 
					
						
							|  |  |  |       pop_text_stack(lvl); | 
					
						
							|  |  |  |       return (unsigned char *)Yap_TermToBuffer(inp->val.t, 0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (IsVarTerm(inp->val.t)) { | 
					
						
							|  |  |  |       Yap_ThrowError(INSTANTIATION_ERROR, inp->val.t, NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (IsPairOrNilTerm(inp->val.t)) { | 
					
						
							|  |  |  |       if (((inp->type & (YAP_STRING_CODES | YAP_STRING_ATOMS)) == | 
					
						
							|  |  |  | 	   (YAP_STRING_CODES | YAP_STRING_ATOMS))) { | 
					
						
							|  |  |  | 	// Yap_DebugPlWriteln(inp->val.t); | 
					
						
							|  |  |  | 	out = (char *)Yap_ListToBuffer(NULL, inp->val.t, inp PASS_REGS); | 
					
						
							|  |  |  | 	POPRET( out ); | 
					
						
							|  |  |  | 	// this is a term, extract to a sfer, and representation is wide | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (inp->type & YAP_STRING_CODES) { | 
					
						
							|  |  |  | 	// Yap_DebugPlWriteln(inp->val.t); | 
					
						
							|  |  |  | 	out = (char *)Yap_ListOfCodesToBuffer(NULL, inp->val.t, inp PASS_REGS); | 
					
						
							|  |  |  | 	// this is a term, extract to a sfer, and representation is wide | 
					
						
							|  |  |  | 	POPRET( out ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (inp->type & YAP_STRING_ATOMS) { | 
					
						
							|  |  |  | 	// Yap_DebugPlWriteln(inp->val.t); | 
					
						
							|  |  |  | 	out = (char *)Yap_ListOfAtomsToBuffer(NULL, inp->val.t, inp PASS_REGS); | 
					
						
							|  |  |  | 	// this is a term, extract to a buffer, and representation is wide | 
					
						
							|  |  |  | 	POPRET( out ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (IsStringTerm(inp->val.t)) { | 
					
						
							|  |  |  |       if(!(inp->type & (YAP_STRING_STRING))) { | 
					
						
							|  |  |  | 	Yap_ThrowError(type_error(inp), inp->val.t, NULL); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // this is a term, extract to a buffer, and representation is wide | 
					
						
							|  |  |  |       // Yap_DebugPlWriteln(inp->val.t); | 
					
						
							|  |  |  |       const char *s = StringOfTerm(inp->val.t); | 
					
						
							|  |  |  |       if (s[0] == 0) { | 
					
						
							|  |  |  | 	out = Malloc(4); | 
					
						
							|  |  |  | 	memset(out, 0, 4); | 
					
						
							|  |  |  | 	POPRET( out ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (inp->type & YAP_STRING_WITH_BUFFER) { | 
					
						
							|  |  |  | 	pop_text_stack(lvl); | 
					
						
							|  |  |  | 	return (unsigned char *)UStringOfTerm(inp->val.t); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	inp->type |= YAP_STRING_IN_TMP; | 
					
						
							|  |  |  | 	size_t sz = strlen(s); | 
					
						
							|  |  |  | 	out = Malloc(sz + 1); | 
					
						
							|  |  |  | 	strcpy(out, s); | 
					
						
							|  |  |  | 	POPRET( out ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (IsAtomTerm(inp->val.t)) { | 
					
						
							|  |  |  |       if(!(inp->type & (YAP_STRING_ATOM))) { | 
					
						
							|  |  |  | 	Yap_ThrowError(type_error(inp), inp->val.t, NULL); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // this is a term, extract to a buffer, and representation is wide | 
					
						
							|  |  |  |       // Yap_DebugPlWriteln(inp->val.t); | 
					
						
							|  |  |  |       Atom at = AtomOfTerm(inp->val.t); | 
					
						
							|  |  |  |       if (RepAtom(at)->UStrOfAE[0] == '\0') { | 
					
						
							|  |  |  | 	out = Malloc(4); | 
					
						
							|  |  |  | 	memset(out, 0, 4); | 
					
						
							|  |  |  | 	POPRET( out ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (inp->type & YAP_STRING_WITH_BUFFER) { | 
					
						
							|  |  |  | 	pop_text_stack(lvl); | 
					
						
							|  |  |  | 	return at->UStrOfAE; | 
					
						
							|  |  |  |       } else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  size_t sz = strlen(at->StrOfAE); | 
					
						
							|  |  |  | 	  out = Malloc(sz + 1); | 
					
						
							|  |  |  | 	  strcpy(out, at->StrOfAE); | 
					
						
							|  |  |  | 	  POPRET( out ); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (inp->type & YAP_STRING_INT && IsIntegerTerm(inp->val.t)) { | 
					
						
							|  |  |  |       if(!(inp->type & (YAP_STRING_INT))) { | 
					
						
							|  |  |  | 	Yap_ThrowError(type_error(inp), inp->val.t, NULL); | 
					
						
							|  |  |  |       }    // ASCII, so both LATIN1 and UTF-8 | 
					
						
							|  |  |  |       // Yap_DebugPlWriteln(inp->val.t); | 
					
						
							|  |  |  |       out = Malloc(2 * MaxTmp(PASS_REGS1)); | 
					
						
							|  |  |  |       if (snprintf(out, MaxTmp(PASS_REGS1) - 1, Int_FORMAT, | 
					
						
							|  |  |  | 		   IntegerOfTerm(inp->val.t)) < 0) { | 
					
						
							|  |  |  | 	AUX_ERROR(inp->val.t, 2 * MaxTmp(PASS_REGS1), out, char); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       POPRET( out ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (inp->type & YAP_STRING_FLOAT && IsFloatTerm(inp->val.t)) { | 
					
						
							|  |  |  |       if(!(inp->type & (YAP_STRING_FLOAT))) { | 
					
						
							|  |  |  | 	Yap_ThrowError(type_error(inp), inp->val.t, NULL); | 
					
						
							|  |  |  |       }    // ASCII, so both LATIN1 and UTF-8 | 
					
						
							|  |  |  |       out = Malloc(2 * MaxTmp(PASS_REGS1)); | 
					
						
							|  |  |  |       if (!Yap_FormatFloat(FloatOfTerm(inp->val.t), &out, 1024)) { | 
					
						
							|  |  |  | 	pop_text_stack(lvl); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       POPRET(out); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | #if USE_GMP | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     if ( IsBigIntTerm(inp->val.t)) { | 
					
						
							|  |  |  |       if(!(inp->type & (YAP_STRING_BIG))) { | 
					
						
							|  |  |  | 	Yap_ThrowError(type_error(inp), inp->val.t, NULL); | 
					
						
							|  |  |  |       }    // ASCII, so both LATIN1 and UTF-8 | 
					
						
							|  |  |  |       // Yap_DebugPlWriteln(inp->val.t); | 
					
						
							|  |  |  |       out = Malloc(MaxTmp()); | 
					
						
							|  |  |  |       if (!Yap_mpz_to_string(Yap_BigIntOfTerm(inp->val.t), out, MaxTmp() - 1, | 
					
						
							|  |  |  | 			     10)) { | 
					
						
							|  |  |  | 	AUX_ERROR(inp->val.t, MaxTmp(PASS_REGS1), out, char); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       POPRET(out); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   pop_text_stack(lvl); | 
					
						
							|  |  |  |   Yap_ThrowError(type_error(inp), inp->val.t, NULL); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static Term write_strings(unsigned char *s0, seq_tv_t *out USES_REGS) { | 
					
						
							|  |  |  |   size_t min = 0, max = strlen((char *)s0); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (out->type & (YAP_STRING_NCHARS | YAP_STRING_TRUNC)) { | 
					
						
							|  |  |  |     if (out->type & YAP_STRING_NCHARS) | 
					
						
							|  |  |  |       min = out->max; | 
					
						
							|  |  |  |     if (out->type & YAP_STRING_TRUNC && out->max < max) { | 
					
						
							|  |  |  |       max = out->max; | 
					
						
							|  |  |  |       s0[max] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   char *s = (char *)s0; | 
					
						
							|  |  |  |   Term t = init_tstring(PASS_REGS1); | 
					
						
							|  |  |  |   LOCAL_TERM_ERROR(t, 2 * max); | 
					
						
							|  |  |  |   unsigned char *buf = buf_from_tstring(HR); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (max == 0) | 
					
						
							|  |  |  |     buf[0] = '\0'; | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     strcpy((char *)buf, s); | 
					
						
							|  |  |  |   if (max + 1 < min) { | 
					
						
							|  |  |  |     LOCAL_TERM_ERROR(t, 2 * min); | 
					
						
							|  |  |  |     memset(buf + min, '\0', max); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     buf += min; | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     buf += max + 1; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   close_tstring(buf PASS_REGS); | 
					
						
							|  |  |  |   out->val.t = t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return out->val.t; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static Term write_atoms(void *s0, seq_tv_t *out USES_REGS) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   Term t = AbsPair(HR); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   char *s1 = (char *)s0; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   size_t sz = 0; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   size_t max = strlen(s1); | 
					
						
							|  |  |  |   if (s1[0] == '\0') { | 
					
						
							|  |  |  |     out->val.t = TermNil; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     return TermNil; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (out->type & (YAP_STRING_NCHARS | YAP_STRING_TRUNC)) { | 
					
						
							|  |  |  |     if (out->type & YAP_STRING_TRUNC && out->max < max) | 
					
						
							|  |  |  |       max = out->max; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   unsigned char *s = s0, *lim = s + strnlen((char *)s, max); | 
					
						
							|  |  |  |   unsigned char *cp = s; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   unsigned char w[10]; | 
					
						
							|  |  |  |   int wp = 0; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   LOCAL_TERM_ERROR(t, 2 * (lim - s)); | 
					
						
							|  |  |  |   while (cp < lim && *cp) { | 
					
						
							|  |  |  |     utf8proc_int32_t chr; | 
					
						
							|  |  |  |     CELL *cl; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     s += get_utf8(s, -1, &chr); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     if (chr == '\0') { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       w[0] = '\0'; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     wp = put_utf8(w, chr); | 
					
						
							|  |  |  |     w[wp] = '\0'; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     cl = HR; | 
					
						
							|  |  |  |     HR += 2; | 
					
						
							|  |  |  |     cl[0] = MkAtomTerm(Yap_ULookupAtom(w)); | 
					
						
							|  |  |  |     cl[1] = AbsPair(HR); | 
					
						
							|  |  |  |     sz++; | 
					
						
							|  |  |  |     if (sz == max) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (out->type & YAP_STRING_DIFF) { | 
					
						
							|  |  |  |     if (sz == 0) | 
					
						
							|  |  |  |       t = out->dif; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       HR[-1] = Globalize(out->dif PASS_REGS); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (sz == 0) | 
					
						
							|  |  |  |       t = TermNil; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       HR[-1] = TermNil; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   out->val.t = t; | 
					
						
							|  |  |  |   return (t); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static Term write_codes(void *s0, seq_tv_t *out USES_REGS) { | 
					
						
							|  |  |  |   Term t; | 
					
						
							|  |  |  |   size_t sz = strlen(s0); | 
					
						
							|  |  |  |   if (sz == 0) { | 
					
						
							|  |  |  |     if (out->type & YAP_STRING_DIFF) { | 
					
						
							|  |  |  |       out->val.t = Globalize(out->dif PASS_REGS); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       out->val.t = TermNil; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return out->val.t; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   unsigned char *s = s0, *lim = s + sz; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   unsigned char *cp = s; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   t = AbsPair(HR); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   LOCAL_TERM_ERROR(t, 2 * (lim - s)); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   t = AbsPair(HR); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   while (*cp) { | 
					
						
							|  |  |  |     utf8proc_int32_t chr; | 
					
						
							|  |  |  |     CELL *cl; | 
					
						
							|  |  |  |     cp += get_utf8(cp, -1, &chr); | 
					
						
							|  |  |  |     if (chr == '\0') | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     cl = HR; | 
					
						
							|  |  |  |     HR += 2; | 
					
						
							|  |  |  |     cl[0] = MkIntegerTerm(chr); | 
					
						
							|  |  |  |     cl[1] = AbsPair(HR); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (sz == 0) { | 
					
						
							|  |  |  |     HR[-1] = Globalize(out->dif PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     HR[-1] = TermNil; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   out->val.t = t; | 
					
						
							|  |  |  |   return (t); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static Atom write_atom(void *s0, seq_tv_t *out USES_REGS) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   unsigned char *s = s0; | 
					
						
							|  |  |  |   int32_t ch; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (s[0] == '\0') { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     return Yap_LookupAtom(""); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   size_t leng = strlen(s0); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   if (strlen_utf8(s0) <= leng) { | 
					
						
							|  |  |  |     return Yap_LookupAtom(s0); | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     size_t n = get_utf8(s, -1, &ch); | 
					
						
							|  |  |  |     unsigned char *buf = Malloc(n + 1); | 
					
						
							|  |  |  |     memmove(buf, s0, n + 1); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     return Yap_ULookupAtom(buf); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | void *write_buffer(unsigned char *s0, seq_tv_t *out USES_REGS) { | 
					
						
							|  |  |  |   int l = push_text_stack(); | 
					
						
							|  |  |  |   size_t leng = strlen((char *)s0); | 
					
						
							|  |  |  |   size_t min = 0, max = leng; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   if (out->enc == ENC_ISO_UTF8) { | 
					
						
							|  |  |  |     if (out->val.uc == NULL) { // this should always be the case | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       out->val.uc = Malloc(leng + 1); | 
					
						
							|  |  |  |       strcpy(out->val.c, (char *)s0); | 
					
						
							|  |  |  |     } else if (out->val.uc != s0) { | 
					
						
							|  |  |  |       out->val.c = Malloc(leng + 1); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       strcpy(out->val.c, (char *)s0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else if (out->enc == ENC_ISO_LATIN1) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     unsigned char *s = s0; | 
					
						
							|  |  |  |     unsigned char *cp = s; | 
					
						
							|  |  |  |     unsigned char *buf = out->val.uc; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     if (!buf) { | 
					
						
							|  |  |  |       pop_text_stack(l); | 
					
						
							|  |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     while (*cp) { | 
					
						
							|  |  |  |       utf8proc_int32_t chr; | 
					
						
							|  |  |  |       int off = get_utf8(cp, -1, &chr); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       if (off <= 0 || chr > 255) { | 
					
						
							|  |  |  | 	pop_text_stack(l); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       if (off == max) | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 	break; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       cp += off; | 
					
						
							|  |  |  |       *buf++ = chr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (max >= min) | 
					
						
							|  |  |  |       *buf++ = '\0'; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       while (max < min) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 	utf8proc_int32_t chr; | 
					
						
							|  |  |  | 	max++; | 
					
						
							|  |  |  | 	cp += get_utf8(cp, -1, &chr); | 
					
						
							|  |  |  | 	*buf++ = chr; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |   } else if (out->enc == ENC_WCHAR) { | 
					
						
							|  |  |  |     unsigned char *s = s0, *lim = s + (max = strnlen((char *)s0, max)); | 
					
						
							|  |  |  |     unsigned char *cp = s; | 
					
						
							|  |  |  |     wchar_t *buf0, *buf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     buf = buf0 = out->val.w; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     if (!buf) { | 
					
						
							|  |  |  |       pop_text_stack(l); | 
					
						
							|  |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     while (*cp && cp < lim) { | 
					
						
							|  |  |  |       utf8proc_int32_t chr; | 
					
						
							|  |  |  |       cp += get_utf8(cp, -1, &chr); | 
					
						
							|  |  |  |       *buf++ = chr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (max >= min) | 
					
						
							|  |  |  |       *buf++ = '\0'; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       while (max < min) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 	utf8proc_int32_t chr; | 
					
						
							|  |  |  | 	max++; | 
					
						
							|  |  |  | 	cp += get_utf8(cp, -1, &chr); | 
					
						
							|  |  |  | 	*buf++ = chr; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     *buf = '\0'; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     // no other encodings are supported. | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     pop_text_stack(l); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   out->val.c = pop_output_text_stack(l, out->val.c); | 
					
						
							|  |  |  |   return out->val.c; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static size_t write_length(const unsigned char *s0, seq_tv_t *out USES_REGS) { | 
					
						
							|  |  |  |   return strlen_utf8(s0); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static Term write_number(unsigned char *s, seq_tv_t *out, | 
					
						
							|  |  |  | 			 bool error_on USES_REGS) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   Term t; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   LOCAL_delay = !error_on; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   t = Yap_StringToNumberTerm((char *)s, &out->enc, error_on); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   LOCAL_delay = false; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   return t; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static Term string_to_term(void *s, seq_tv_t *out USES_REGS) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   Term o; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   yap_error_descriptor_t *new_error = malloc(sizeof(yap_error_descriptor_t)); | 
					
						
							|  |  |  |   bool mdnew = Yap_pushErrorContext(true, new_error); | 
					
						
							|  |  |  |   o = out->val.t = Yap_BufferToTerm(s, TermNil); | 
					
						
							|  |  |  |   Yap_popErrorContext(mdnew, true); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   return o; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | bool write_Text(unsigned char *inp, seq_tv_t *out USES_REGS) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   /* we know what the term is */ | 
					
						
							|  |  |  |   if (out->type == 0) { | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (LOCAL_Error_TYPE) { | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   if (out->type & (YAP_STRING_INT | YAP_STRING_FLOAT | YAP_STRING_BIG)) { | 
					
						
							|  |  |  |     if ((out->val.t = write_number( | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 				   inp, out, !(out->type & YAP_STRING_ATOM) PASS_REGS)) != 0L) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       // Yap_DebugPlWriteln(out->val.t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!(out->type & YAP_STRING_ATOM)) | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (out->type & (YAP_STRING_ATOM)) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     if ((out->val.a = write_atom(inp, out PASS_REGS)) != NIL) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       Atom at = out->val.a; | 
					
						
							|  |  |  |       if (at && (out->type & YAP_STRING_OUTPUT_TERM)) | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 	out->val.t = MkAtomTerm(at); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       // Yap_DebugPlWriteln(out->val.t); | 
					
						
							|  |  |  |       return at != NIL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (out->type & YAP_STRING_DATUM) { | 
					
						
							|  |  |  |     if ((out->val.t = string_to_term(inp, out PASS_REGS)) != 0L) | 
					
						
							|  |  |  |       return out->val.t != 0; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   switch (out->type & YAP_TYPE_MASK) { | 
					
						
							|  |  |  |   case YAP_STRING_CHARS: { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     void *room = write_buffer(inp, out PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     // printf("%s\n", out->val.c); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     return room != NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   case YAP_STRING_WCHARS: { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     void *room = write_buffer(inp, out PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     // printf("%S\n", out->val.w); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     return room != NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   case YAP_STRING_STRING: | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     out->val.t = write_strings(inp, out PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     // Yap_DebugPlWriteln(out->val.t); | 
					
						
							|  |  |  |     return out->val.t != 0; | 
					
						
							|  |  |  |   case YAP_STRING_ATOMS: | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     out->val.t = write_atoms(inp, out PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     // Yap_DebugPlWriteln(out->val.t); | 
					
						
							|  |  |  |     return out->val.t != 0; | 
					
						
							|  |  |  |   case YAP_STRING_CODES: | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     out->val.t = write_codes(inp, out PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     // Yap_DebugPlWriteln(out->val.t); | 
					
						
							|  |  |  |     return out->val.t != 0; | 
					
						
							|  |  |  |   case YAP_STRING_LENGTH: | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     out->val.l = write_length(inp, out PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     // printf("s\n",out->val.l); | 
					
						
							|  |  |  |     return out->val.l != (size_t)(-1); | 
					
						
							|  |  |  |   case YAP_STRING_ATOM: | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     out->val.a = write_atom(inp, out PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     // Yap_DebugPlWriteln(out->val.t); | 
					
						
							|  |  |  |     return out->val.a != NULL; | 
					
						
							|  |  |  |   case YAP_STRING_INT | YAP_STRING_FLOAT | YAP_STRING_BIG: | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     out->val.t = write_number(inp, out, true PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     // Yap_DebugPlWriteln(out->val.t); | 
					
						
							|  |  |  |     return out->val.t != 0; | 
					
						
							|  |  |  |   default: { return true; } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static size_t upcase(void *s0, seq_tv_t *out USES_REGS) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   unsigned char *s = s0; | 
					
						
							|  |  |  |   while (*s) { | 
					
						
							|  |  |  |     // assumes the two code have always the same size; | 
					
						
							|  |  |  |     utf8proc_int32_t chr; | 
					
						
							|  |  |  |     get_utf8(s, -1, &chr); | 
					
						
							|  |  |  |     chr = utf8proc_toupper(chr); | 
					
						
							|  |  |  |     s += put_utf8(s, chr); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static size_t downcase(void *s0, seq_tv_t *out USES_REGS) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   unsigned char *s = s0; | 
					
						
							|  |  |  |   while (*s) { | 
					
						
							|  |  |  |     // assumes the two code have always the same size; | 
					
						
							|  |  |  |     utf8proc_int32_t chr; | 
					
						
							|  |  |  |     get_utf8(s, -1, &chr); | 
					
						
							|  |  |  |     chr = utf8proc_tolower(chr); | 
					
						
							|  |  |  |     s += put_utf8(s, chr); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Yap_CVT_Text(seq_tv_t *inp, seq_tv_t *out USES_REGS) { | 
					
						
							|  |  |  |   unsigned char *buf; | 
					
						
							|  |  |  |   bool rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   //printf(stderr, "[ %d ", n++)    ; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   if (inp->type & (YAP_STRING_TERM|YAP_STRING_ATOM|YAP_STRING_ATOMS_CODES | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   |YAP_STRING_STRING)) | 
					
						
							|  |  |  |   //Yap_DebugPlWriteln(inp->val.t); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   else if (inp->type & YAP_STRING_WCHARS) fprintf(stderr,"S %S\n", inp->val | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   .w); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   else  fprintf(stderr,"s %s\n", inp->val.c); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   */ | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   //  cnt++; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   int l = push_text_stack(); | 
					
						
							|  |  |  |   buf = Yap_readText(inp PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   if (!buf) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     pop_text_stack(l); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     return 0L; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (buf[0]) { | 
					
						
							|  |  |  |     size_t leng = strlen_utf8(buf); | 
					
						
							|  |  |  |     if (out->type & (YAP_STRING_NCHARS | YAP_STRING_TRUNC)) { | 
					
						
							|  |  |  |       if (out->max < leng) { | 
					
						
							|  |  |  | 	const unsigned char *ptr = skip_utf8(buf, out->max); | 
					
						
							|  |  |  | 	size_t diff = (ptr - buf); | 
					
						
							|  |  |  | 	char *nbuf = Malloc(diff + 1); | 
					
						
							|  |  |  | 	memmove(nbuf, buf, diff); | 
					
						
							|  |  |  | 	nbuf[diff] = '\0'; | 
					
						
							|  |  |  | 	leng = diff; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       // else if (out->type & YAP_STRING_NCHARS && | 
					
						
							|  |  |  |       // const unsigned char *ptr = skip_utf8(buf) | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (out->type & (YAP_STRING_UPCASE | YAP_STRING_DOWNCASE)) { | 
					
						
							|  |  |  |       if (out->type & YAP_STRING_UPCASE) { | 
					
						
							|  |  |  | 	if (!upcase(buf, out)) { | 
					
						
							|  |  |  | 	  pop_text_stack(l); | 
					
						
							|  |  |  | 	  return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (out->type & YAP_STRING_DOWNCASE) { | 
					
						
							|  |  |  | 	if (!downcase(buf, out)) { | 
					
						
							|  |  |  | 	  pop_text_stack(l); | 
					
						
							|  |  |  | 	  return false; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   rc = write_Text(buf, out PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   /*    fprintf(stderr, " -> "); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 	if (!rc) fprintf(stderr, "NULL"); | 
					
						
							|  |  |  | 	else if (out->type & | 
					
						
							|  |  |  | 	(YAP_STRING_TERM|YAP_STRING_ATOMS_CODES | 
					
						
							|  |  |  | 	|YAP_STRING_STRING)) //Yap_DebugPlWrite(out->val.t); | 
					
						
							|  |  |  | 	else if (out->type & | 
					
						
							|  |  |  | 	YAP_STRING_ATOM) //Yap_DebugPlWriteln(MkAtomTerm(out->val.a)); | 
					
						
							|  |  |  | 	else if (out->type & YAP_STRING_WCHARS) fprintf(stderr, "%S", | 
					
						
							|  |  |  | 	out->val.w); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	fprintf(stderr, "%s", out->val.c); | 
					
						
							|  |  |  | 	fprintf(stderr, "\n]\n"); */ | 
					
						
							|  |  |  |   pop_text_stack(l); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static unsigned char *concat(int n, void *sv[] USES_REGS) { | 
					
						
							|  |  |  |   void *buf; | 
					
						
							|  |  |  |   unsigned char *buf0; | 
					
						
							|  |  |  |   size_t room = 0; | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < n; i++) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     char *s = sv[i]; | 
					
						
							|  |  |  |     if (s[0]) | 
					
						
							|  |  |  |       room += strlen(s); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   buf = Malloc(room + 1); | 
					
						
							|  |  |  |   buf0 = buf; | 
					
						
							|  |  |  |   for (i = 0; i < n; i++) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     char *s = sv[i]; | 
					
						
							|  |  |  |     if (!s[0]) | 
					
						
							|  |  |  |       continue; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | #if _WIN32 || defined(__ANDROID__) | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     strcpy(buf, s); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     buf = (char *)buf + strlen(buf); | 
					
						
							|  |  |  | #else | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     buf = stpcpy(buf, s); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | #endif | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return buf0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | static void *slice(size_t min, size_t max, const unsigned char *buf USES_REGS) { | 
					
						
							|  |  |  |   unsigned char *nbuf = BaseMalloc((max - min) * 4 + 1); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   const unsigned char *ptr = skip_utf8(buf, min); | 
					
						
							|  |  |  |   unsigned char *nptr = nbuf; | 
					
						
							|  |  |  |   utf8proc_int32_t chr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (min++ < max) { | 
					
						
							|  |  |  |     ptr += get_utf8(ptr, -1, &chr); | 
					
						
							|  |  |  |     nptr += put_utf8(nptr, chr); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   nptr[0] = '\0'; | 
					
						
							|  |  |  |   return nbuf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Out must be an atom or a string | 
					
						
							|  |  |  | bool Yap_Concat_Text(int tot, seq_tv_t inp[], seq_tv_t *out USES_REGS) { | 
					
						
							|  |  |  |   void **bufv; | 
					
						
							|  |  |  |   unsigned char *buf; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   int i, j; | 
					
						
							|  |  |  |   // int lvl = push_text_stack(); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   bufv = Malloc(tot * sizeof(unsigned char *)); | 
					
						
							|  |  |  |   if (!bufv) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     // pop_text_stack(lvl); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   for (i = 0, j = 0; i < tot; i++) { | 
					
						
							|  |  |  |     //  inp[j].type |= YAP_STRING_WITH_BUFFER; | 
					
						
							|  |  |  |     unsigned char *nbuf = Yap_readText(inp + i PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!nbuf) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       // pop_text_stack(lvl); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     //      if (!nbuf[0]) | 
					
						
							|  |  |  |     //	continue; | 
					
						
							|  |  |  |     bufv[j++] = nbuf; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (j == 0) { | 
					
						
							|  |  |  |     buf = Malloc(8); | 
					
						
							|  |  |  |     memset(buf, 0, 4); | 
					
						
							|  |  |  |   } else if (j == 1) { | 
					
						
							|  |  |  |     buf = bufv[0]; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     buf = concat(tot, bufv PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   bool rc = write_Text(buf, out PASS_REGS); | 
					
						
							|  |  |  |   // pop_text_stack( lvl ); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | bool Yap_Splice_Text(int n, size_t cuts[], seq_tv_t *inp, | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 		     seq_tv_t outv[] USES_REGS) { | 
					
						
							|  |  |  |   int lvl = push_text_stack(); | 
					
						
							|  |  |  |   const unsigned char *buf; | 
					
						
							|  |  |  |   size_t b_l, u_l; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   inp->type |= YAP_STRING_IN_TMP; | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   buf = Yap_readText(inp PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   if (!buf) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     pop_text_stack(lvl); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   b_l = strlen((char *)buf); | 
					
						
							|  |  |  |   if (b_l == 0) { | 
					
						
							|  |  |  |     pop_text_stack(lvl); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   u_l = strlen_utf8(buf); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   if (!cuts) { | 
					
						
							|  |  |  |     if (n == 2) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       size_t b_l0, b_l1, u_l0, u_l1; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       unsigned char *buf0, *buf1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (outv[0].val.t) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 	buf0 = Yap_readText(outv PASS_REGS); | 
					
						
							|  |  |  | 	if (!buf0) { | 
					
						
							|  |  |  | 	  return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	b_l0 = strlen((const char *)buf0); | 
					
						
							|  |  |  | 	if (memcmp(buf, buf0, b_l0) != 0) { | 
					
						
							|  |  |  | 	  pop_text_stack(lvl); | 
					
						
							|  |  |  | 	  return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	u_l0 = strlen_utf8(buf0); | 
					
						
							|  |  |  | 	u_l1 = u_l - u_l0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	b_l1 = b_l - b_l0; | 
					
						
							|  |  |  | 	buf1 = slice(u_l0, u_l, buf PASS_REGS); | 
					
						
							|  |  |  | 	b_l1 = strlen((const char *)buf1); | 
					
						
							|  |  |  | 	bool rc = write_Text(buf1, outv + 1 PASS_REGS); | 
					
						
							|  |  |  | 	pop_text_stack(lvl); | 
					
						
							|  |  |  | 	if (!rc) { | 
					
						
							|  |  |  | 	  return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } else /* if (outv[1].val.t) */ { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  | 	buf1 = Yap_readText(outv + 1 PASS_REGS); | 
					
						
							|  |  |  | 	if (!buf1) { | 
					
						
							|  |  |  | 	  pop_text_stack(lvl); | 
					
						
							|  |  |  | 	  return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	b_l1 = strlen((char *)buf1); | 
					
						
							|  |  |  | 	u_l1 = strlen_utf8(buf1); | 
					
						
							|  |  |  | 	b_l0 = b_l - b_l1; | 
					
						
							|  |  |  | 	u_l0 = u_l - u_l1; | 
					
						
							|  |  |  | 	if (memcmp(skip_utf8((const unsigned char *)buf, b_l0), buf1, b_l1) != | 
					
						
							|  |  |  | 	    0) { | 
					
						
							|  |  |  | 	  pop_text_stack(lvl); | 
					
						
							|  |  |  | 	  return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	buf0 = slice(0, u_l0, buf PASS_REGS); | 
					
						
							|  |  |  | 	buf0 = pop_output_text_stack(lvl, buf0); | 
					
						
							|  |  |  | 	bool rc = write_Text(buf0, outv PASS_REGS); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   int i, next; | 
					
						
							|  |  |  |   for (i = 0; i < n; i++) { | 
					
						
							|  |  |  |     if (i == 0) | 
					
						
							|  |  |  |       next = 0; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       next = cuts[i - 1]; | 
					
						
							|  |  |  |     if (i > 0 && cuts[i] == 0) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     void *bufi = slice(next, cuts[i], buf PASS_REGS); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     bufi = pop_output_text_stack(lvl, bufi); | 
					
						
							|  |  |  |     if (!write_Text(bufi, outv + i PASS_REGS)) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   pop_text_stack(lvl); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Convert from a predicate structure to an UTF-8 string of the form | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * module:name/arity. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The result is in very volatile memory. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param s        the buffer | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @return the temporary string | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const char *Yap_PredIndicatorToUTF8String(PredEntry *ap) { | 
					
						
							|  |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     Atom at; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   arity_t arity = 0; | 
					
						
							|  |  |  |   Functor f; | 
					
						
							|  |  |  |   char *s, *smax, *s0; | 
					
						
							|  |  |  |   s = s0 = malloc(1024); | 
					
						
							|  |  |  |   smax = s + 1024; | 
					
						
							|  |  |  |   Term tmod = ap->ModuleOfPred; | 
					
						
							|  |  |  |   if (tmod) { | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |     char *sn = Yap_AtomToUTF8Text(AtomOfTerm(tmod)); | 
					
						
							|  |  |  |     stpcpy(s, sn); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     if (smax - s > 1) { | 
					
						
							|  |  |  |       strcat(s, ":"); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     s++; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (smax - s > strlen("prolog:")) { | 
					
						
							|  |  |  |       s = strcpy(s, "prolog:"); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   // follows the actual functor | 
					
						
							|  |  |  |   if (ap->ModuleOfPred == IDB_MODULE) { | 
					
						
							|  |  |  |     if (ap->PredFlags & NumberDBPredFlag) { | 
					
						
							|  |  |  |       Int key = ap->src.IndxId; | 
					
						
							|  |  |  |       snprintf(s, smax - s, "%" PRIdPTR, key); | 
					
						
							|  |  |  |       return LOCAL_FileNameBuf; | 
					
						
							|  |  |  |     } else if (ap->PredFlags & AtomDBPredFlag) { | 
					
						
							|  |  |  |       at = (Atom)(ap->FunctorOfPred); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |       if (!stpcpy(s, Yap_AtomToUTF8Text(at))) | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     } else { | 
					
						
							|  |  |  |       f = ap->FunctorOfPred; | 
					
						
							|  |  |  |       at = NameOfFunctor(f); | 
					
						
							|  |  |  |       arity = ArityOfFunctor(f); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     arity = ap->ArityOfPE; | 
					
						
							|  |  |  |     if (arity) { | 
					
						
							|  |  |  |       at = NameOfFunctor(ap->FunctorOfPred); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       at = (Atom)(ap->FunctorOfPred); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   if (!stpcpy(s, Yap_AtomToUTF8Text(at))) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   s += strlen(s); | 
					
						
							|  |  |  |   snprintf(s, smax - s, "/%" PRIdPTR, arity); | 
					
						
							|  |  |  |   return s0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Convert from a text buffer (8-bit) to a term that has the same type as | 
					
						
							|  |  |  |  * _Tguide_ | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  ≈* @param s        the buffer | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |  ≈ * @param tguide   the guide | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |  ≈  * @return the term | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | Term Yap_MkTextTerm(const char *s, int guide USES_REGS) { | 
					
						
							|  |  |  |   if (guide == YAP_STRING_ATOM) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     return MkAtomTerm(Yap_LookupAtom(s)); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   } else if (guide == YAP_STRING_STRING) { | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |     return MkStringTerm(s); | 
					
						
							| 
									
										
										
										
											2018-07-21 01:56:48 +01:00
										 |  |  |   } else if (guide == YAP_STRING_ATOMS) { | 
					
						
							|  |  |  |     return Yap_CharsToListOfAtoms(s, ENC_ISO_UTF8 PASS_REGS); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     return Yap_CharsToListOfCodes(s, ENC_ISO_UTF8 PASS_REGS); | 
					
						
							| 
									
										
										
										
											2017-11-11 03:05:36 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } |