| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | /*************************************************************************
 | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *	 YAP Prolog 							 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *	Yap Prolog was developed at NCCUP - Universidade do Porto	 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | ************************************************************************** | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * File:		arith1.c						 * | 
					
						
							|  |  |  | * Last rev:								 * | 
					
						
							|  |  |  | * mods:									 * | 
					
						
							|  |  |  | * comments:	bignum support through gmp				 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *************************************************************************/ | 
					
						
							|  |  |  | #ifdef SCCS
 | 
					
						
							|  |  |  | static char     SccsId[] = "%W% %G%"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Yap.h"
 | 
					
						
							|  |  |  | #include "Yatom.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-19 22:42:47 +01:00
										 |  |  | #if HAVE_STRING_H
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-22 04:09:33 -07:00
										 |  |  | #include "YapHeap.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "eval.h"
 | 
					
						
							|  |  |  | #include "alloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Term | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_MkBigIntTerm(MP_INT *big) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |   Int nlimbs; | 
					
						
							| 
									
										
										
										
											2008-11-28 15:54:46 +00:00
										 |  |  |   MP_INT *dst = (MP_INT *)(H+2); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |   CELL *ret = H; | 
					
						
							| 
									
										
										
										
											2012-02-06 15:10:48 +00:00
										 |  |  |   Int bytes; | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-16 02:57:52 +00:00
										 |  |  |   if (mpz_fits_slong_p(big)) { | 
					
						
							| 
									
										
										
										
											2006-02-01 13:28:57 +00:00
										 |  |  |     long int out = mpz_get_si(big); | 
					
						
							|  |  |  |     return MkIntegerTerm((Int)out); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:25:45 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-02-06 15:10:48 +00:00
										 |  |  |   //  bytes = big->_mp_alloc * sizeof(mp_limb_t);
 | 
					
						
							|  |  |  |   //  nlimbs = ALIGN_YAPTYPE(bytes,CELL)/CellSize;
 | 
					
						
							|  |  |  |   // this works, but it shouldn't need to do this...
 | 
					
						
							|  |  |  |   nlimbs = big->_mp_alloc; | 
					
						
							|  |  |  |   bytes = nlimbs*sizeof(CELL); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:25:45 +00:00
										 |  |  |   if (nlimbs > (ASP-ret)-1024) { | 
					
						
							|  |  |  |     return TermNil; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   H[0] = (CELL)FunctorBigInt; | 
					
						
							| 
									
										
										
										
											2008-11-28 15:54:46 +00:00
										 |  |  |   H[1] = BIG_INT; | 
					
						
							| 
									
										
										
										
											2006-01-02 02:25:45 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   dst->_mp_size = big->_mp_size; | 
					
						
							| 
									
										
										
										
											2012-02-06 15:10:48 +00:00
										 |  |  |   dst->_mp_alloc = nlimbs*(CellSize/sizeof(mp_limb_t)); | 
					
						
							|  |  |  |   memmove((void *)(dst+1), (const void *)(big->_mp_d), bytes); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:25:45 +00:00
										 |  |  |   H = (CELL *)(dst+1)+nlimbs; | 
					
						
							| 
									
										
										
										
											2011-07-21 02:24:21 -07:00
										 |  |  |   H[0] = EndSpecials; | 
					
						
							|  |  |  |   H++; | 
					
						
							|  |  |  |   return AbsAppl(ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:25:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | MP_INT * | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_BigIntOfTerm(Term t) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-11-28 15:54:46 +00:00
										 |  |  |   MP_INT *new = (MP_INT *)(RepAppl(t)+2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   new->_mp_d = (mp_limb_t *)(new+1); | 
					
						
							|  |  |  |   return(new); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-27 12:24:15 +01:00
										 |  |  | Term | 
					
						
							|  |  |  | Yap_MkBigRatTerm(MP_RAT *big) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2010-05-27 12:24:15 +01:00
										 |  |  |   Int nlimbs; | 
					
						
							|  |  |  |   MP_INT *dst = (MP_INT *)(H+2); | 
					
						
							|  |  |  |   MP_INT *num = mpq_numref(big); | 
					
						
							|  |  |  |   MP_INT *den = mpq_denref(big); | 
					
						
							|  |  |  |   MP_RAT *rat; | 
					
						
							|  |  |  |   CELL *ret = H; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (mpz_cmp_si(den, 1) == 0) | 
					
						
							|  |  |  |     return Yap_MkBigIntTerm(num); | 
					
						
							|  |  |  |   if ((num->_mp_alloc+den->_mp_alloc)*(sizeof(mp_limb_t)/CellSize) > (ASP-ret)-1024) { | 
					
						
							|  |  |  |     return TermNil; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   H[0] = (CELL)FunctorBigInt; | 
					
						
							|  |  |  |   H[1] = BIG_RATIONAL; | 
					
						
							| 
									
										
										
										
											2010-05-28 09:53:56 +01:00
										 |  |  |   dst->_mp_size = 0; | 
					
						
							| 
									
										
										
										
											2010-05-27 12:24:15 +01:00
										 |  |  |   rat = (MP_RAT *)(dst+1); | 
					
						
							|  |  |  |   rat->_mp_num._mp_size = num->_mp_size; | 
					
						
							|  |  |  |   rat->_mp_num._mp_alloc = num->_mp_alloc; | 
					
						
							|  |  |  |   nlimbs = (num->_mp_alloc)*(sizeof(mp_limb_t)/CellSize); | 
					
						
							|  |  |  |   memmove((void *)(rat+1), (const void *)(num->_mp_d), nlimbs*CellSize); | 
					
						
							|  |  |  |   rat->_mp_den._mp_size = den->_mp_size; | 
					
						
							|  |  |  |   rat->_mp_den._mp_alloc = den->_mp_alloc; | 
					
						
							|  |  |  |   H = (CELL *)(rat+1)+nlimbs; | 
					
						
							|  |  |  |   nlimbs = (den->_mp_alloc)*(sizeof(mp_limb_t)/CellSize); | 
					
						
							|  |  |  |   memmove((void *)(H), (const void *)(den->_mp_d), nlimbs*CellSize); | 
					
						
							|  |  |  |   H += nlimbs; | 
					
						
							| 
									
										
										
										
											2010-05-28 09:53:56 +01:00
										 |  |  |   dst->_mp_alloc = (H-(CELL *)(dst+1)); | 
					
						
							| 
									
										
										
										
											2010-05-27 12:24:15 +01:00
										 |  |  |   H[0] = EndSpecials; | 
					
						
							|  |  |  |   H++; | 
					
						
							|  |  |  |   return AbsAppl(ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MP_RAT * | 
					
						
							|  |  |  | Yap_BigRatOfTerm(Term t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   MP_RAT *new = (MP_RAT *)(RepAppl(t)+2+sizeof(MP_INT)/sizeof(CELL)); | 
					
						
							|  |  |  |   mp_limb_t *nt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   nt = new->_mp_num._mp_d = (mp_limb_t *)(new+1); | 
					
						
							|  |  |  |   nt += new->_mp_num._mp_alloc; | 
					
						
							|  |  |  |   new->_mp_den._mp_d = nt; | 
					
						
							|  |  |  |   return new; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-28 09:53:56 +01:00
										 |  |  | Term  | 
					
						
							|  |  |  | Yap_RatTermToApplTerm(Term t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Term ts[2]; | 
					
						
							|  |  |  |   MP_RAT *rat = Yap_BigRatOfTerm(t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ts[0] =  Yap_MkBigIntTerm(mpq_numref(rat)); | 
					
						
							|  |  |  |   ts[1] =  Yap_MkBigIntTerm(mpq_denref(rat)); | 
					
						
							|  |  |  |   return Yap_MkApplTerm(FunctorRDiv,2,ts); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-22 04:09:33 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Term | 
					
						
							|  |  |  | Yap_AllocExternalDataInStack(CELL tag, size_t bytes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CACHE_REGS | 
					
						
							|  |  |  |   Int nlimbs; | 
					
						
							|  |  |  |   MP_INT *dst = (MP_INT *)(H+2); | 
					
						
							|  |  |  |   CELL *ret = H; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   nlimbs = ALIGN_YAPTYPE(bytes,CELL)/CellSize; | 
					
						
							|  |  |  |   if (nlimbs > (ASP-ret)-1024) { | 
					
						
							|  |  |  |     return TermNil; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   H[0] = (CELL)FunctorBigInt; | 
					
						
							|  |  |  |   H[1] = tag; | 
					
						
							|  |  |  |   dst->_mp_size = 0; | 
					
						
							|  |  |  |   dst->_mp_alloc = nlimbs; | 
					
						
							|  |  |  |   H = (CELL *)(dst+1)+nlimbs; | 
					
						
							|  |  |  |   H[0] = EndSpecials; | 
					
						
							|  |  |  |   H++; | 
					
						
							|  |  |  |   if (tag != EXTERNAL_BLOB) { | 
					
						
							|  |  |  |     TrailTerm(TR) = AbsPair(ret); | 
					
						
							|  |  |  |     TR++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return AbsAppl(ret); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-27 12:24:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-22 04:09:33 -07:00
										 |  |  | int Yap_CleanOpaqueVariable(CELL *pt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CELL blob_info, blob_tag; | 
					
						
							|  |  |  |   MP_INT *blobp; | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   /* sanity checking */ | 
					
						
							|  |  |  |   if (pt[0] != (CELL)FunctorBigInt) { | 
					
						
							|  |  |  |     Yap_Error(SYSTEM_ERROR, TermNil, "CleanOpaqueVariable bad call"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-07-22 04:09:33 -07:00
										 |  |  |   blob_tag = pt[1]; | 
					
						
							|  |  |  |   if (blob_tag < USER_BLOB_START || | 
					
						
							|  |  |  |       blob_tag >= USER_BLOB_END) { | 
					
						
							|  |  |  |     Yap_Error(SYSTEM_ERROR, AbsAppl(pt), "clean opaque: bad blob with tag " UInt_FORMAT ,blob_tag); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   blob_info = blob_tag - USER_BLOB_START; | 
					
						
							|  |  |  |   if (!GLOBAL_OpaqueHandlers) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   blobp = (MP_INT *)(pt+2); | 
					
						
							| 
									
										
										
										
											2011-07-22 04:21:21 -07:00
										 |  |  |   if (!GLOBAL_OpaqueHandlers[blob_info].fail_handler) | 
					
						
							|  |  |  |     return TRUE; | 
					
						
							| 
									
										
										
										
											2011-07-22 04:09:33 -07:00
										 |  |  |   return (GLOBAL_OpaqueHandlers[blob_info].fail_handler)((void *)(blobp+1)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-27 16:50:14 +01:00
										 |  |  | Opaque_CallOnWrite | 
					
						
							|  |  |  | Yap_blob_write_handler_from_slot(Int slot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-09-15 15:40:47 +01:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2011-07-27 16:50:14 +01:00
										 |  |  |   CELL blob_info, blob_tag; | 
					
						
							| 
									
										
										
										
											2011-09-15 15:40:47 +01:00
										 |  |  |   Term t = Yap_GetFromSlot(slot PASS_REGS); | 
					
						
							| 
									
										
										
										
											2011-07-27 16:50:14 +01:00
										 |  |  |   CELL *pt = RepAppl(t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   /* sanity checking */ | 
					
						
							|  |  |  |   if (pt[0] != (CELL)FunctorBigInt) { | 
					
						
							|  |  |  |     Yap_Error(SYSTEM_ERROR, TermNil, "CleanOpaqueVariable bad call"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   blob_tag = pt[1]; | 
					
						
							|  |  |  |   if (blob_tag < USER_BLOB_START || | 
					
						
							|  |  |  |       blob_tag >= USER_BLOB_END) { | 
					
						
							|  |  |  |     Yap_Error(SYSTEM_ERROR, AbsAppl(pt), "clean opaque: bad blob with tag " UInt_FORMAT ,blob_tag); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   blob_info = blob_tag - USER_BLOB_START; | 
					
						
							| 
									
										
										
										
											2011-12-13 10:01:51 +00:00
										 |  |  |   if (!GLOBAL_OpaqueHandlers) { | 
					
						
							| 
									
										
										
										
											2011-07-27 16:50:14 +01:00
										 |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2011-12-13 10:01:51 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-07-27 16:50:14 +01:00
										 |  |  |   return GLOBAL_OpaqueHandlers[blob_info].write_handler; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-13 10:01:51 +00:00
										 |  |  | Opaque_CallOnGCMark | 
					
						
							|  |  |  | Yap_blob_gc_mark_handler(Term t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CELL blob_info, blob_tag; | 
					
						
							|  |  |  |   CELL *pt = RepAppl(t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   /* sanity checking */ | 
					
						
							|  |  |  |   if (pt[0] != (CELL)FunctorBigInt) { | 
					
						
							|  |  |  |     Yap_Error(SYSTEM_ERROR, TermNil, "CleanOpaqueVariable bad call"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   blob_tag = pt[1]; | 
					
						
							|  |  |  |   if (blob_tag < USER_BLOB_START || | 
					
						
							|  |  |  |       blob_tag >= USER_BLOB_END) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   blob_info = blob_tag - USER_BLOB_START; | 
					
						
							|  |  |  |   if (!GLOBAL_OpaqueHandlers) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   return GLOBAL_OpaqueHandlers[blob_info].gc_mark_handler; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-13 19:14:33 +01:00
										 |  |  | Opaque_CallOnGCRelocate | 
					
						
							|  |  |  | Yap_blob_gc_relocate_handler(Term t) | 
					
						
							| 
									
										
										
										
											2011-12-13 10:01:51 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   CELL blob_info, blob_tag; | 
					
						
							|  |  |  |   CELL *pt = RepAppl(t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   /* sanity checking */ | 
					
						
							|  |  |  |   if (pt[0] != (CELL)FunctorBigInt) { | 
					
						
							|  |  |  |     Yap_Error(SYSTEM_ERROR, TermNil, "CleanOpaqueVariable bad call"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   blob_tag = pt[1]; | 
					
						
							|  |  |  |   if (blob_tag < USER_BLOB_START || | 
					
						
							|  |  |  |       blob_tag >= USER_BLOB_END) { | 
					
						
							|  |  |  |     Yap_Error(SYSTEM_ERROR, AbsAppl(pt), "clean opaque: bad blob with tag " UInt_FORMAT ,blob_tag); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   blob_info = blob_tag - USER_BLOB_START; | 
					
						
							|  |  |  |   if (!GLOBAL_OpaqueHandlers) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2011-12-13 19:14:33 +01:00
										 |  |  |   return GLOBAL_OpaqueHandlers[blob_info].gc_relocate_handler; | 
					
						
							| 
									
										
										
										
											2011-12-13 10:01:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-27 16:50:14 +01:00
										 |  |  | extern Int Yap_blob_tag_from_slot(Int slot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-09-15 15:40:47 +01:00
										 |  |  |   CACHE_REGS | 
					
						
							|  |  |  |   Term t = Yap_GetFromSlot(slot PASS_REGS); | 
					
						
							| 
									
										
										
										
											2011-07-27 16:50:14 +01:00
										 |  |  |   CELL *pt = RepAppl(t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   /* sanity checking */ | 
					
						
							|  |  |  |   if (pt[0] != (CELL)FunctorBigInt) { | 
					
						
							|  |  |  |     Yap_Error(SYSTEM_ERROR, TermNil, "CleanOpaqueVariable bad call"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return pt[1]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void * | 
					
						
							|  |  |  | Yap_blob_info_from_slot(Int slot) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-09-15 15:40:47 +01:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2011-07-27 16:50:14 +01:00
										 |  |  |   MP_INT *blobp; | 
					
						
							| 
									
										
										
										
											2011-09-15 15:40:47 +01:00
										 |  |  |   Term t = Yap_GetFromSlot(slot PASS_REGS); | 
					
						
							| 
									
										
										
										
											2011-07-27 16:50:14 +01:00
										 |  |  |   CELL *pt = RepAppl(t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   /* sanity checking */ | 
					
						
							|  |  |  |   if (pt[0] != (CELL)FunctorBigInt) { | 
					
						
							|  |  |  |     Yap_Error(SYSTEM_ERROR, TermNil, "CleanOpaqueVariable bad call"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   if (!GLOBAL_OpaqueHandlers) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   blobp = (MP_INT *)(pt+2); | 
					
						
							|  |  |  |   return (void *)(blobp+1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-20 15:10:18 +00:00
										 |  |  | Term | 
					
						
							|  |  |  | Yap_MkULLIntTerm(YAP_ULONG_LONG n) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-11-16 02:01:09 +00:00
										 |  |  | #if __GNUC__ && USE_GMP
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     mpz_t new; | 
					
						
							| 
									
										
										
										
											2005-11-15 02:05:49 +00:00
										 |  |  |     char tmp[256]; | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     Term t; | 
					
						
							| 
									
										
										
										
											2005-11-15 02:05:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-10 03:03:03 -06:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  |     snprintf(tmp,256,"%I64u",n); | 
					
						
							|  |  |  | #elif HAVE_SNPRINTF
 | 
					
						
							| 
									
										
										
										
											2005-11-16 01:55:03 +00:00
										 |  |  |     snprintf(tmp,256,"%llu",n); | 
					
						
							| 
									
										
										
										
											2005-11-15 02:05:49 +00:00
										 |  |  | #else    
 | 
					
						
							|  |  |  |     sprintf(tmp,"%llu",n); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     /* try to scan it as a bignum */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     mpz_init_set_str (new, tmp, 10); | 
					
						
							| 
									
										
										
										
											2005-11-15 02:05:49 +00:00
										 |  |  |     if (mpz_fits_slong_p(new)) { | 
					
						
							|  |  |  |       return MkIntegerTerm(mpz_get_si(new)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     t = Yap_MkBigIntTerm(new); | 
					
						
							| 
									
										
										
										
											2006-01-18 15:34:54 +00:00
										 |  |  |     mpz_clear(new); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     return t; | 
					
						
							| 
									
										
										
										
											2003-03-20 15:10:18 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     return MkIntegerTerm(n); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | static Int  | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | p_is_bignum( USES_REGS1 ) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   Term t = Deref(ARG1); | 
					
						
							| 
									
										
										
										
											2008-11-28 15:54:46 +00:00
										 |  |  |   return( | 
					
						
							| 
									
										
										
										
											2010-05-28 09:53:56 +01:00
										 |  |  | 	 IsNonVarTerm(t) &&  | 
					
						
							|  |  |  | 	 IsApplTerm(t) &&  | 
					
						
							|  |  |  | 	 FunctorOfTerm(t) == FunctorBigInt && | 
					
						
							|  |  |  | 	 RepAppl(t)[1] == BIG_INT | 
					
						
							|  |  |  | 	 ); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2007-08-02 17:22:00 +00:00
										 |  |  |   return FALSE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int  | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | p_has_bignums( USES_REGS1 ) | 
					
						
							| 
									
										
										
										
											2007-08-02 17:22:00 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return FALSE; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-28 09:53:56 +01:00
										 |  |  | static Int  | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | p_is_rational( USES_REGS1 ) | 
					
						
							| 
									
										
										
										
											2010-05-28 09:53:56 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   if (IsVarTerm(t)) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   if (IsIntTerm(t)) | 
					
						
							|  |  |  |     return TRUE; | 
					
						
							|  |  |  |   if (IsApplTerm(t)) { | 
					
						
							|  |  |  |     Functor f = FunctorOfTerm(t); | 
					
						
							|  |  |  |     CELL *pt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (f == FunctorLongInt) | 
					
						
							|  |  |  |       return TRUE; | 
					
						
							|  |  |  |     if (f != FunctorBigInt) | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     pt = RepAppl(t); | 
					
						
							|  |  |  |     return (  pt[1] == BIG_RATIONAL || pt[1] == BIG_INT ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return FALSE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int  | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  | p_rational( USES_REGS1 ) | 
					
						
							| 
									
										
										
										
											2010-05-28 09:53:56 +01:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   Functor f; | 
					
						
							|  |  |  |   CELL *pt; | 
					
						
							|  |  |  |   MP_RAT *rat; | 
					
						
							|  |  |  |   Term t1, t2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(t)) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   if (!IsApplTerm(t)) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   f = FunctorOfTerm(t); | 
					
						
							|  |  |  |   if (f != FunctorBigInt) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   pt = RepAppl(t); | 
					
						
							|  |  |  |   if (pt[1] != BIG_RATIONAL) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   rat = Yap_BigRatOfTerm(t); | 
					
						
							|  |  |  |   while ((t1 = Yap_MkBigIntTerm(mpq_numref(rat))) == TermNil || | 
					
						
							|  |  |  | 	 (t2 = Yap_MkBigIntTerm(mpq_denref(rat))) == TermNil) { | 
					
						
							|  |  |  |     UInt size = | 
					
						
							|  |  |  |       (mpq_numref(rat)->_mp_alloc)*(sizeof(mp_limb_t)/CellSize) + | 
					
						
							|  |  |  |       (mpq_denref(rat)->_mp_alloc)*(sizeof(mp_limb_t)/CellSize); | 
					
						
							|  |  |  |     if (!Yap_gcl(size, 3, ENV, P)) { | 
					
						
							| 
									
										
										
										
											2011-05-23 16:19:47 +01:00
										 |  |  |       Yap_Error(OUT_OF_STACK_ERROR, t, LOCAL_ErrorMessage); | 
					
						
							| 
									
										
										
										
											2010-05-28 09:53:56 +01:00
										 |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return  | 
					
						
							|  |  |  |     Yap_unify(ARG2, t1) && | 
					
						
							|  |  |  |     Yap_unify(ARG3, t2); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return FALSE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-11 23:21:23 +00:00
										 |  |  | int | 
					
						
							|  |  |  | Yap_IsStringTerm(Term t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CELL fl; | 
					
						
							|  |  |  |   if (IsVarTerm(t)) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   if (!IsApplTerm(t)) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   if (FunctorOfTerm(t) != FunctorBigInt) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fl = RepAppl(t)[1]; | 
					
						
							|  |  |  |   return fl == BLOB_STRING || fl == BLOB_WIDE_STRING; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | Yap_IsWideStringTerm(Term t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CELL fl; | 
					
						
							|  |  |  |   if (IsVarTerm(t)) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   if (!IsApplTerm(t)) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   if (FunctorOfTerm(t) != FunctorBigInt) | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fl = RepAppl(t)[1]; | 
					
						
							|  |  |  |   return fl == BLOB_WIDE_STRING; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  | Term | 
					
						
							|  |  |  | Yap_MkBlobStringTerm(const char *s, size_t len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  |   CELL *ret = H; | 
					
						
							|  |  |  |   size_t sz; | 
					
						
							|  |  |  |   MP_INT *dst = (MP_INT *)(H+2); | 
					
						
							|  |  |  |   blob_string_t *sp; | 
					
						
							|  |  |  |   size_t siz; | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |   char *dest; | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   sz = strlen(s); | 
					
						
							|  |  |  |   if (len > 0 && sz > len) sz = len; | 
					
						
							|  |  |  |   if (len/sizeof(CELL) > (ASP-ret)-1024) { | 
					
						
							|  |  |  |     return TermNil; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   H[0] = (CELL)FunctorBigInt; | 
					
						
							|  |  |  |   H[1] = BLOB_STRING; | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |   siz = ALIGN_YAPTYPE((len+1+sizeof(blob_string_t)),CELL); | 
					
						
							| 
									
										
										
										
											2011-06-14 08:56:21 +01:00
										 |  |  |   dst->_mp_size = 0L; | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |   dst->_mp_alloc = siz/sizeof(mp_limb_t); | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  |   sp = (blob_string_t *)(dst+1); | 
					
						
							|  |  |  |   sp->len = sz; | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |   dest = (char *)(sp+1); | 
					
						
							|  |  |  |   strncpy(dest, s, sz); | 
					
						
							|  |  |  |   dest[sz] = '\0'; | 
					
						
							|  |  |  |   H += (siz + 2*sizeof(CELL)+sizeof(MP_INT)+sizeof(Functor))/sizeof(CELL); | 
					
						
							|  |  |  |   H[-1] = EndSpecials; | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  |   return AbsAppl(ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Term | 
					
						
							|  |  |  | Yap_MkBlobWideStringTerm(const wchar_t *s, size_t len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 16:02:55 +00:00
										 |  |  |   CACHE_REGS | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  |   CELL *ret = H; | 
					
						
							|  |  |  |   size_t sz; | 
					
						
							|  |  |  |   MP_INT *dst = (MP_INT *)(H+2); | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |   blob_string_t *sp = (blob_string_t *)(dst+1); | 
					
						
							| 
									
										
										
										
											2011-06-12 17:22:45 +01:00
										 |  |  |   size_t siz, i = 0; | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |   H[0] = (CELL)FunctorBigInt; | 
					
						
							|  |  |  |   dst->_mp_size = 0L; | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  |   sz = wcslen(s); | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |   if (len > 0 && sz > len) { | 
					
						
							|  |  |  |     sz = len; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if ((len/sizeof(CELL)) > (ASP-ret)-1024) { | 
					
						
							|  |  |  |     return TermNil; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-06-12 17:22:45 +01:00
										 |  |  |   while (i < sz) { | 
					
						
							|  |  |  |     if (s[i++] >= 255) break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (i == sz) { | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |     /* we have a standard ascii string */ | 
					
						
							| 
									
										
										
										
											2011-06-12 17:22:45 +01:00
										 |  |  |     char *target; | 
					
						
							|  |  |  |     size_t i = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     H[1] = BLOB_STRING; | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |     siz = ALIGN_YAPTYPE((sz+1+sizeof(blob_string_t)),CELL); | 
					
						
							|  |  |  |     dst->_mp_alloc = siz/sizeof(mp_limb_t); | 
					
						
							| 
									
										
										
										
											2011-06-12 17:22:45 +01:00
										 |  |  |     sp->len = sz; | 
					
						
							|  |  |  |     target = (char *)(sp+1); | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |     for (i = 0 ; i < sz; i++) { | 
					
						
							| 
									
										
										
										
											2011-06-12 17:22:45 +01:00
										 |  |  |       target[i] = s[i]; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |     target[sz] = '\0'; | 
					
						
							|  |  |  |     H += (siz+2*sizeof(CELL)+sizeof(MP_INT)+sizeof(Functor))/sizeof(CELL); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     wchar_t * target; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     H[1] = BLOB_WIDE_STRING; | 
					
						
							|  |  |  |     siz = ALIGN_YAPTYPE((sz+1)*sizeof(wchar_t)+sizeof(blob_string_t),CELL); | 
					
						
							|  |  |  |     dst->_mp_alloc = siz/sizeof(mp_limb_t); | 
					
						
							|  |  |  |     sp->len = sz; | 
					
						
							|  |  |  |     target = (wchar_t *)(sp+1);  | 
					
						
							|  |  |  |     wcsncpy(target, s, sz); | 
					
						
							|  |  |  |     target[sz] = '\0'; | 
					
						
							|  |  |  |     H += (siz + 2*sizeof(CELL)+sizeof(MP_INT)+sizeof(Functor))/sizeof(CELL); | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-06-21 15:11:07 +01:00
										 |  |  |   H[-1] = EndSpecials; | 
					
						
							| 
									
										
										
										
											2010-06-19 00:38:49 +01:00
										 |  |  |   return AbsAppl(ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | Yap_BlobStringOfTerm(Term t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   blob_string_t *new = (blob_string_t *)(RepAppl(t)+2+sizeof(MP_INT)/sizeof(CELL)); | 
					
						
							|  |  |  |   return (char *)(new+1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | wchar_t * | 
					
						
							|  |  |  | Yap_BlobWideStringOfTerm(Term t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   blob_string_t *new = (blob_string_t *)(RepAppl(t)+2+sizeof(MP_INT)/sizeof(CELL)); | 
					
						
							|  |  |  |   return (wchar_t *)(new+1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | Yap_BlobStringOfTermAndLength(Term t, size_t *sp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   blob_string_t *new = (blob_string_t *)(RepAppl(t)+2+sizeof(MP_INT)/sizeof(CELL)); | 
					
						
							|  |  |  |   *sp = new->len; | 
					
						
							|  |  |  |   return (char *)(new+1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_InitBigNums(void) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-10-19 18:10:48 +01:00
										 |  |  |   Yap_InitCPred("$has_bignums", 0, p_has_bignums, SafePredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("$bignum", 1, p_is_bignum, SafePredFlag); | 
					
						
							| 
									
										
										
										
											2010-05-28 09:53:56 +01:00
										 |  |  |   Yap_InitCPred("rational", 3, p_rational, 0); | 
					
						
							|  |  |  |   Yap_InitCPred("rational", 1, p_is_rational, SafePredFlag); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } |