| 
									
										
										
										
											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:		arith2.c						 * | 
					
						
							|  |  |  | * Last rev:								 * | 
					
						
							|  |  |  | * mods:									 * | 
					
						
							|  |  |  | * comments:	arithmetical expression evaluation			 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *************************************************************************/ | 
					
						
							|  |  |  | #ifdef SCCS
 | 
					
						
							|  |  |  | static char     SccsId[] = "%W% %G%"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * This file implements unary arithmetic operations in YAP | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Yap.h"
 | 
					
						
							|  |  |  | #include "Yatom.h"
 | 
					
						
							|  |  |  | #include "Heap.h"
 | 
					
						
							|  |  |  | #include "eval.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define E_FUNC   blob_type
 | 
					
						
							|  |  |  | #define E_ARGS   , arith_retptr o
 | 
					
						
							|  |  |  | #define USE_E_ARGS   , o
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-12 15:51:23 +00:00
										 |  |  | #define TMP_BIG()   ((o)->big)
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | #define RINT(v)     (o)->Int = (v); return(long_int_e)
 | 
					
						
							|  |  |  | #define RFLOAT(v)   (o)->dbl = (v); return(double_e)
 | 
					
						
							|  |  |  | #define RBIG(v)     return(big_int_e)
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #define RERROR()    return(db_ref_e)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | #define ArithIEval(t,v)     Yap_Eval(t,v)
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | inline static Functor | 
					
						
							|  |  |  | AritFunctorOfTerm(Term t) { | 
					
						
							|  |  |  |   if (IsVarTerm(t)) { | 
					
						
							|  |  |  |     return(FunctorPortray); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (IsApplTerm(t)) { | 
					
						
							|  |  |  |     return(FunctorOfTerm(t)); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (IsIntTerm(t)) | 
					
						
							|  |  |  |       return(FunctorLongInt); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       return(FunctorDBRef); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline static Term | 
					
						
							|  |  |  | EvalToTerm(blob_type f, union arith_ret *res) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   switch (f) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     return(MkIntegerTerm(res->Int)); | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     return(MkFloatTerm(res->dbl)); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       Term t = Yap_MkBigIntTerm(res->big); | 
					
						
							| 
									
										
										
										
											2006-01-18 15:34:54 +00:00
										 |  |  |       mpz_clear(res->big); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       return t; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     return(TermNil); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef blob_type (*f_binexp)(Term, Term, arith_retptr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct init_bin_eval { | 
					
						
							|  |  |  |   char          *OpName; | 
					
						
							|  |  |  |   f_binexp        f; | 
					
						
							|  |  |  | } InitBinEntry; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "arith2.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |   modulus mod (* now follows ISO standard *) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | */ | 
					
						
							|  |  |  | static E_FUNC | 
					
						
							|  |  |  | p_mod(Term t1, Term t2 E_ARGS) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Functor f1 = AritFunctorOfTerm(t1), f2; | 
					
						
							|  |  |  |   blob_type bt1, bt2; | 
					
						
							|  |  |  |   union arith_ret v1, v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (BlobOfFunctor(f1)) { | 
					
						
							|  |  |  |   case (CELL)long_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case (CELL)long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	Int i1 = IntegerOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	Int mod; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	if (i2 == 0) goto zero_divisor; | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mod = i1%i2; | 
					
						
							|  |  |  | 	if (mod && (mod ^ i2) < 0) | 
					
						
							|  |  |  | 	  mod += i2; | 
					
						
							|  |  |  | 	RINT(mod); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     case (CELL)double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, t2, "mod/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case (CELL)big_int_e: | 
					
						
							|  |  |  |       /* I know the term is much larger, so: */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 	Int i1 = IntegerOfTerm(t1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mpz_init_set_si(new, i1); | 
					
						
							|  |  |  | 	mpz_fdiv_r(new, new, Yap_BigIntOfTerm(t2)); | 
					
						
							|  |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.Int = IntegerOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = long_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case (CELL)double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_INTEGER, t1, "mod/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case (CELL)big_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* modulo between bignum and integer */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_t tmp; | 
					
						
							|  |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	if (i2 == 0) goto zero_divisor; | 
					
						
							|  |  |  | 	mpz_init(new); | 
					
						
							|  |  |  | 	mpz_init_set_si(tmp, i2); | 
					
						
							|  |  |  | 	mpz_fdiv_r(new, Yap_BigIntOfTerm(t1), tmp); | 
					
						
							|  |  |  | 	mpz_clear(tmp); | 
					
						
							|  |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case (CELL)big_int_e: | 
					
						
							|  |  |  |       /* two bignums */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_init(new); | 
					
						
							|  |  |  | 	mpz_fdiv_r(new, Yap_BigIntOfTerm(t1), Yap_BigIntOfTerm(t2)); | 
					
						
							|  |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       Yap_Error(TYPE_ERROR_INTEGER, t2, "mod/2"); | 
					
						
							|  |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       mpz_init_set(v1.big,Yap_BigIntOfTerm(t1)); | 
					
						
							|  |  |  |       bt1 = big_int_e; | 
					
						
							|  |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |     bt1 = Yap_Eval(t1, &v1); | 
					
						
							|  |  |  |     /* don't know anything about second */ | 
					
						
							|  |  |  |     bt2 = Yap_Eval(t2, &v2); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* second case, no need no evaluation */ | 
					
						
							|  |  |  |   switch (bt1) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = v1.Int; | 
					
						
							|  |  |  | 	Int i2 = v2.Int; | 
					
						
							|  |  |  | 	Int mod; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	if (i2 == 0) goto zero_divisor; | 
					
						
							|  |  |  | 	mod = i1%i2; | 
					
						
							|  |  |  | 	if (mod && (mod ^ i2) < 0) | 
					
						
							|  |  |  | 	  mod += i2; | 
					
						
							|  |  |  | 	RINT(mod); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v2.dbl), "mod/2"); | 
					
						
							|  |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case (CELL)big_int_e: | 
					
						
							|  |  |  |       /* I know the term is much larger, so: */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	mpz_t tmp; | 
					
						
							|  |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MPZ_SET(new, v2.big); | 
					
						
							|  |  |  | 	mpz_init_set_si(tmp, v1.Int); | 
					
						
							|  |  |  | 	mpz_fdiv_r(new, tmp, new); | 
					
						
							|  |  |  | 	mpz_clear(tmp); | 
					
						
							|  |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* Yap_Error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case double_e: | 
					
						
							| 
									
										
										
										
											2006-01-02 23:19:10 +00:00
										 |  |  | #if USE_GMP
 | 
					
						
							|  |  |  |     if (bt2 == big_int_e) | 
					
						
							|  |  |  |       mpz_clear(v2.big); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v1.dbl), "mod/2"); | 
					
						
							|  |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case (CELL)big_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* big mod integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	mpz_t tmp; | 
					
						
							|  |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (v2.Int == 0) goto zero_divisor; | 
					
						
							|  |  |  | 	MPZ_SET(new,v1.big); | 
					
						
							|  |  |  | 	mpz_init_set_si(tmp, v2.Int); | 
					
						
							|  |  |  | 	mpz_fdiv_r(new, new, tmp); | 
					
						
							|  |  |  | 	mpz_clear(tmp); | 
					
						
							|  |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* big // float */ | 
					
						
							|  |  |  |       Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v2.dbl), "mod/2"); | 
					
						
							|  |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     case (CELL)big_int_e: | 
					
						
							|  |  |  |       /* big * big */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	mpz_fdiv_r(new, new, v2.big); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error  */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |  zero_divisor: | 
					
						
							|  |  |  |   Yap_Error(EVALUATION_ERROR_ZERO_DIVISOR, t2, "X is mod 0"); | 
					
						
							|  |  |  |   /* make GCC happy */ | 
					
						
							|  |  |  |   P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |   RERROR(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   remainder rem (* now follows ISO standard *) | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | static E_FUNC | 
					
						
							|  |  |  | p_rem(Term t1, Term t2 E_ARGS) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Functor f1 = AritFunctorOfTerm(t1), f2; | 
					
						
							|  |  |  |   blob_type bt1, bt2; | 
					
						
							|  |  |  |   union arith_ret v1, v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (BlobOfFunctor(f1)) { | 
					
						
							|  |  |  |   case (CELL)long_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case (CELL)long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = IntegerOfTerm(t1); | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	Int mod; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (i2 == 0) goto zero_divisor; | 
					
						
							|  |  |  | 	mod = i1%i2; | 
					
						
							|  |  |  | 	RINT(i1%i2); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case (CELL)double_e: | 
					
						
							|  |  |  |       Yap_Error(TYPE_ERROR_INTEGER, t2, "mod/2"); | 
					
						
							|  |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case (CELL)big_int_e: | 
					
						
							|  |  |  |       /* I know the term is much larger, so: */ | 
					
						
							|  |  |  |       RINT(IntegerOfTerm(t1)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.Int = IntegerOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = long_int_e; | 
					
						
							|  |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case (CELL)double_e: | 
					
						
							|  |  |  |     Yap_Error(TYPE_ERROR_INTEGER, t1, "mod/2"); | 
					
						
							|  |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case (CELL)big_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* modulo between bignum and integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	mpz_t tmp; | 
					
						
							|  |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (i2 == 0) goto zero_divisor; | 
					
						
							|  |  |  | 	mpz_init(new); | 
					
						
							|  |  |  | 	mpz_init_set_si(tmp, i2); | 
					
						
							|  |  |  | 	mpz_tdiv_r(new, Yap_BigIntOfTerm(t1), tmp); | 
					
						
							|  |  |  | 	mpz_clear(tmp); | 
					
						
							|  |  |  | 	RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     case (CELL)big_int_e: | 
					
						
							|  |  |  |       /* two bignums */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_init(new); | 
					
						
							|  |  |  | 	mpz_tdiv_r(new, Yap_BigIntOfTerm(t1), Yap_BigIntOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, t2, "mod/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       mpz_init_set(v1.big,Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       bt1 = big_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt1 = Yap_Eval(t1, &v1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* don't know anything about second */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* second case, no need no evaluation */ | 
					
						
							|  |  |  |   switch (bt1) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = v1.Int; | 
					
						
							|  |  |  | 	Int i2 = v2.Int; | 
					
						
							|  |  |  | 	Int mod; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (i2 == 0) goto zero_divisor; | 
					
						
							|  |  |  | 	mod = i1%i2; | 
					
						
							|  |  |  | 	RINT(mod); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v2.dbl), "mod/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case (CELL)big_int_e: | 
					
						
							|  |  |  |       /* Cool */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       mpz_clear(v2.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       RINT(v1.Int); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       /* Yap_Error */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case double_e: | 
					
						
							| 
									
										
										
										
											2006-01-02 23:19:10 +00:00
										 |  |  | #if USE_GMP
 | 
					
						
							|  |  |  |     if (bt2 == big_int_e) | 
					
						
							|  |  |  |       mpz_clear(v2.big); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v1.dbl), "mod/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case (CELL)big_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* big mod integer */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	mpz_t tmp; | 
					
						
							|  |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	if (v2.Int == 0) goto zero_divisor; | 
					
						
							|  |  |  | 	MPZ_SET(new,v1.big); | 
					
						
							|  |  |  | 	mpz_init_set_si(tmp, v2.Int); | 
					
						
							|  |  |  | 	mpz_tdiv_r(new, new, tmp); | 
					
						
							|  |  |  | 	mpz_clear(tmp); | 
					
						
							|  |  |  | 	RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* big // float */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v2.dbl), "mod/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     case (CELL)big_int_e: | 
					
						
							|  |  |  |       /* big * big */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	mpz_tdiv_r(new, new, v2.big); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error  */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |  zero_divisor: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   Yap_Error(EVALUATION_ERROR_ZERO_DIVISOR, t2, "X is mod 0"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   /* make GCC happy */ | 
					
						
							|  |  |  |   P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |   RERROR(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  | static Float | 
					
						
							|  |  |  | fdiv_bigint(MP_INT *b1,MP_INT *b2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Float f1 = mpz_get_d(b1); | 
					
						
							|  |  |  |   Float f2 = mpz_get_d(b2); | 
					
						
							|  |  |  |   if (1) { | 
					
						
							|  |  |  |     mpf_t f1,f2; | 
					
						
							|  |  |  |     Float res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     mpf_init(f1); | 
					
						
							|  |  |  |     mpf_init(f2); | 
					
						
							|  |  |  |     mpf_set_z(f1, b1); | 
					
						
							|  |  |  |     mpf_set_z(f2, b2); | 
					
						
							|  |  |  |     mpf_div(f1, f1, f2); | 
					
						
							|  |  |  |     res = mpf_get_d(f1); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     mpf_clear(f1); | 
					
						
							|  |  |  |     mpf_clear(f2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     return(res); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     return(f1/f2); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   Floating point division: / | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | static E_FUNC | 
					
						
							|  |  |  | p_fdiv(Term t1, Term t2 E_ARGS) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Functor f1 = AritFunctorOfTerm(t1), f2; | 
					
						
							|  |  |  |   blob_type bt1, bt2; | 
					
						
							|  |  |  |   union arith_ret v1, v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (BlobOfFunctor(f1)) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* two integers */ | 
					
						
							|  |  |  | 	RFLOAT((((Float)IntegerOfTerm(t1))/(Float)i2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	/* integer, double */ | 
					
						
							|  |  |  | 	Float fl1 = (Float)IntegerOfTerm(t1); | 
					
						
							|  |  |  | 	Float fl2 = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	RFLOAT(fl1/fl2); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case (CELL)big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = IntegerOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Float f2 = mpz_get_d(Yap_BigIntOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2003-02-24 11:01:01 +00:00
										 |  |  | 	RFLOAT(((Float)i1/f2)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.Int = IntegerOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = long_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	RFLOAT(FloatOfTerm(t1)/(Float)i2); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float f2 = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	RFLOAT(FloatOfTerm(t1)/f2); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	RFLOAT(FloatOfTerm(t1)/mpz_get_d(Yap_BigIntOfTerm(t2))); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.dbl = FloatOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = double_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	RFLOAT(mpz_get_d(Yap_BigIntOfTerm(t1))/(Float)i); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* two bignums*/ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       RFLOAT(fdiv_bigint(Yap_BigIntOfTerm(t1),Yap_BigIntOfTerm(t2))); | 
					
						
							|  |  |  |       //      RFLOAT(mpz_get_d(Yap_BigIntOfTerm(t1))/mpz_get_d(Yap_BigIntOfTerm(t2)));
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = FloatOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	RFLOAT(mpz_get_d(Yap_BigIntOfTerm(t1))/dbl); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       mpz_init_set(v1.big,Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       bt1 = big_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt1 = Yap_Eval(t1, &v1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* don't know anything about second */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* second case, no need no evaluation */ | 
					
						
							|  |  |  |   switch (bt1) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       RFLOAT((Float)(((float)v1.Int)/(Float)(v2.Int))); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							|  |  |  |       RFLOAT(v1.Int/v2.dbl); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = v1.Int/mpz_get_d(v2.big); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       /* Yap_Error */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       RFLOAT(v1.dbl/v2.Int); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* float / float */ | 
					
						
							|  |  |  |       RFLOAT(v1.dbl/v2.dbl); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* float / float */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = v1.dbl/mpz_get_d(v2.big); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = mpz_get_d(v1.big)/v2.Int; | 
					
						
							|  |  |  | 	mpz_clear(v1.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case double_e: | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = mpz_get_d(v1.big)/v2.dbl; | 
					
						
							|  |  |  | 	mpz_clear(v1.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* big / big */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = fdiv_bigint(v1.big,v2.big); | 
					
						
							|  |  |  | 	mpz_clear(v1.big); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error  */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if USE_GMP
 | 
					
						
							|  |  |  | #if !HAVE_MPZ_XOR
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | mpz_xor(MP_INT *new, MP_INT *r1, MP_INT *r2) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |   MP_INT *n2, *n3; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |   mpz_new(n2); | 
					
						
							|  |  |  |   mpz_new(n3); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   mpz_ior(new, r1, r2); | 
					
						
							|  |  |  |   mpz_com(n2,  r1); | 
					
						
							|  |  |  |   mpz_and(n2, n2, new); | 
					
						
							|  |  |  |   mpz_com(n3,  r2); | 
					
						
							|  |  |  |   mpz_and(n3, n3, new); | 
					
						
							|  |  |  |   mpz_ior(new, n2, n3); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |   mpz_clear(n2); | 
					
						
							|  |  |  |   mpz_clear(n3); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   xor # | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | static E_FUNC | 
					
						
							|  |  |  | p_xor(Term t1, Term t2 E_ARGS) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Functor f1 = AritFunctorOfTerm(t1), f2; | 
					
						
							|  |  |  |   blob_type bt1, bt2; | 
					
						
							|  |  |  |   union arith_ret v1, v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (BlobOfFunctor(f1)) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       RINT(IntegerOfTerm(t1) ^ IntegerOfTerm(t2)); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, t2, "#/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_init_set_si(new,IntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	mpz_xor(new, new, Yap_BigIntOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.Int = IntegerOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = long_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_INTEGER, t1, "#/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_init_set_si(new,IntegerOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	mpz_xor(new, Yap_BigIntOfTerm(t1), new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* two bignums */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_init_set(new, Yap_BigIntOfTerm(t1)); | 
					
						
							|  |  |  | 	mpz_xor(new, new, Yap_BigIntOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, t2, "#/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       mpz_init_set(v1.big,Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       bt1 = big_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt1 = Yap_Eval(t1, &v1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* don't know anything about second */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* second case, no need no evaluation */ | 
					
						
							|  |  |  |   switch (bt1) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       RINT(v1.Int ^ v2.Int); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v2.dbl), "#/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_init_set_si(new,v1.Int); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	mpz_xor(new, new, v2.big); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v2.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       /* Yap_Error */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case double_e: | 
					
						
							| 
									
										
										
										
											2006-01-02 23:19:10 +00:00
										 |  |  | #if USE_GMP
 | 
					
						
							|  |  |  |     if (bt2 == big_int_e) | 
					
						
							|  |  |  |       mpz_clear(v2.big); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v1.dbl), "#/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* anding a bignum with an integer is easy */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_init_set_si(new,v2.Int); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	mpz_xor(new, v1.big, new); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v1.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* big // float */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v2.dbl), "\\/ /2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* big * big */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	mpz_xor(new, new, v2.big); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error  */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   atan2: arc tangent x/y | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | static E_FUNC | 
					
						
							|  |  |  | p_atan2(Term t1, Term t2 E_ARGS) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Functor f1 = AritFunctorOfTerm(t1), f2; | 
					
						
							|  |  |  |   blob_type bt1, bt2; | 
					
						
							|  |  |  |   union arith_ret v1, v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (BlobOfFunctor(f1)) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* two integers */ | 
					
						
							|  |  |  | 	RFLOAT(atan2(IntegerOfTerm(t1),i2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	/* integer, double */ | 
					
						
							|  |  |  | 	Float fl1 = (Float)IntegerOfTerm(t1); | 
					
						
							|  |  |  | 	Float fl2 = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	RFLOAT(atan2(fl1,fl2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = IntegerOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Float f2 = mpz_get_d(Yap_BigIntOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RFLOAT(atan2(i1,f2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.Int = IntegerOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = long_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	RFLOAT(atan2(FloatOfTerm(t1),i2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float f2 = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	RFLOAT(atan2(FloatOfTerm(t1),f2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	RFLOAT(atan2(FloatOfTerm(t1),mpz_get_d(Yap_BigIntOfTerm(t2)))); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.dbl = FloatOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = double_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	RFLOAT(atan2(mpz_get_d(Yap_BigIntOfTerm(t1)),i)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* two bignums */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       RFLOAT(atan2(mpz_get_d(Yap_BigIntOfTerm(t1)),mpz_get_d(Yap_BigIntOfTerm(t2)))); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = FloatOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	RFLOAT(atan2(mpz_get_d(Yap_BigIntOfTerm(t1)),dbl)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       mpz_init_set(v1.big,Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       bt1 = big_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt1 = Yap_Eval(t1, &v1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* don't know anything about second */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* second case, no need no evaluation */ | 
					
						
							|  |  |  |   switch (bt1) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       RFLOAT(atan2(v1.Int,v2.Int)); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							|  |  |  |       RFLOAT(atan2(v1.Int,v2.dbl)); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = atan2(v1.Int,mpz_get_d(v2.big)); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       /* Yap_Error */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       RFLOAT(atan2(v1.dbl,v2.Int)); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* float / float */ | 
					
						
							|  |  |  |       RFLOAT(atan2(v1.dbl,v2.dbl)); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* float / float */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = atan2(v1.dbl,mpz_get_d(v2.big)); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = atan2(mpz_get_d(v1.big),v2.Int); | 
					
						
							|  |  |  | 	mpz_clear(v1.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case double_e: | 
					
						
							|  |  |  |       /* big / float */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = atan2(mpz_get_d(v1.big),v2.dbl); | 
					
						
							|  |  |  | 	mpz_clear(v1.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* big / big */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = atan2(mpz_get_d(v1.big),mpz_get_d(v2.big)); | 
					
						
							|  |  |  | 	mpz_clear(v1.big); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error  */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   power: x^y | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | static E_FUNC | 
					
						
							|  |  |  | p_power(Term t1, Term t2 E_ARGS) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Functor f1 = AritFunctorOfTerm(t1), f2; | 
					
						
							|  |  |  |   blob_type bt1, bt2; | 
					
						
							|  |  |  |   union arith_ret v1, v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (BlobOfFunctor(f1)) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* two integers */ | 
					
						
							|  |  |  | 	RFLOAT(pow(IntegerOfTerm(t1),i2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	/* integer, double */ | 
					
						
							|  |  |  | 	Float fl1 = (Float)IntegerOfTerm(t1); | 
					
						
							|  |  |  | 	Float fl2 = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	RFLOAT(pow(fl1,fl2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = IntegerOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Float f2 = mpz_get_d(Yap_BigIntOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RFLOAT(pow(i1,f2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.Int = IntegerOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = long_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	RFLOAT(pow(FloatOfTerm(t1),i2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float f2 = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	RFLOAT(pow(FloatOfTerm(t1),f2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	RFLOAT(pow(FloatOfTerm(t1),mpz_get_d(Yap_BigIntOfTerm(t2)))); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.dbl = FloatOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = double_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	RFLOAT(pow(mpz_get_d(Yap_BigIntOfTerm(t1)),i)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* two bignums */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       RFLOAT(pow(mpz_get_d(Yap_BigIntOfTerm(t1)),mpz_get_d(Yap_BigIntOfTerm(t2)))); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = FloatOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	RFLOAT(pow(mpz_get_d(Yap_BigIntOfTerm(t1)),dbl)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       mpz_init_set(v1.big,Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       bt1 = big_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt1 = Yap_Eval(t1, &v1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* don't know anything about second */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* second case, no need no evaluation */ | 
					
						
							|  |  |  |   switch (bt1) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       RFLOAT(pow(v1.Int,v2.Int)); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							|  |  |  |       RFLOAT(pow(v1.Int,v2.dbl)); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = pow(v1.Int,mpz_get_d(v2.big)); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       /* Yap_Error */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       RFLOAT(pow(v1.dbl,v2.Int)); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* float / float */ | 
					
						
							|  |  |  |       RFLOAT(pow(v1.dbl,v2.dbl)); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* float / float */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = pow(v1.dbl,mpz_get_d(v2.big)); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = pow(mpz_get_d(v1.big),v2.Int); | 
					
						
							|  |  |  | 	mpz_clear(v1.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case double_e: | 
					
						
							|  |  |  |       /* big / float */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl =  pow(mpz_get_d(v1.big),v2.dbl); | 
					
						
							|  |  |  | 	mpz_clear(v1.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* big / big */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       { | 
					
						
							|  |  |  | 	Float dbl = pow(mpz_get_d(v1.big),mpz_get_d(v2.big)); | 
					
						
							|  |  |  | 	mpz_clear(v1.big); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							|  |  |  | 	RFLOAT(dbl); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error  */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int | 
					
						
							|  |  |  | gcd(Int m11,Int m21) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* Blankinship algorithm, provided by Miguel Filgueiras */ | 
					
						
							|  |  |  |   Int m12=1, m22=0, k; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (m11>0 && m21>0) | 
					
						
							|  |  |  |     if (m11<m21) { | 
					
						
							|  |  |  |       k = m21/m11;  m21 -= k*m11;  m22 -= k*m12; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       k=m11/m21;  m11 -= k*m21;  m12 -= k*m22; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (m11<0 || m21<0) {		/* overflow? */ | 
					
						
							|  |  |  | /*    Oflow = 1; */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_Error(EVALUATION_ERROR_INT_OVERFLOW, MkIntegerTerm(m11), | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  "gcd/2 with %d and %d", m11, m21); | 
					
						
							|  |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     return(1); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (m11)  return(m11); | 
					
						
							|  |  |  |   return(m21); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef GCD_MULT
 | 
					
						
							|  |  |  | Int gcdmult(Int m11,Int m21,Int *pm11)	/* *pm11 gets multiplier of m11 */ | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Int m12=1, m22=0, k; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (m11 && m21) | 
					
						
							|  |  |  |     if (m11<m21) { | 
					
						
							|  |  |  |       k = m21/m11;  m21 -= k*m11;  m22 -= k*m12; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       k=m11/m21;  m11 -= k*m21;  m12 -= k*m22; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (m11<0 || m21<0) {		/* overflow? */ | 
					
						
							|  |  |  | /*    Oflow = 1; */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_Error(EVALUATION_ERROR_INT_OVERFLOW, MkIntegerTerm(m11), | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	  "gcdmult/2 with %d and %d", m11, m21); | 
					
						
							|  |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     return(1); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (m11) { | 
					
						
							|  |  |  |     *pm11 = m12;  return(m11); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   *pm11 = m22; | 
					
						
							|  |  |  |   return(m21); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   module gcd | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | static E_FUNC | 
					
						
							|  |  |  | p_gcd(Term t1, Term t2 E_ARGS) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Functor f1 = AritFunctorOfTerm(t1), f2; | 
					
						
							|  |  |  |   blob_type bt1, bt2; | 
					
						
							|  |  |  |   union arith_ret v1, v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (BlobOfFunctor(f1)) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = IntegerOfTerm(t1), i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	i1 = (i1 >= 0 ? i1 : -i1); | 
					
						
							|  |  |  | 	i2 = (i2 >= 0 ? i2 : -i2); | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  | 	RINT(gcd(i1,i2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, t2, "gcd/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* I know the term is much larger, so: */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (i > 0) { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	  RINT(mpz_gcd_ui(NULL,Yap_BigIntOfTerm(t2),i)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} else if (i == 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  RINT(0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	  RINT(mpz_gcd_ui(NULL,Yap_BigIntOfTerm(t2),-i)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.Int = IntegerOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = long_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_INTEGER, t1, "gcd/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* modulo between bignum and integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (i > 0) { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	  RINT(mpz_gcd_ui(NULL,Yap_BigIntOfTerm(t1),i)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} else if (i == 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  RINT(0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	  RINT(mpz_gcd_ui(NULL,Yap_BigIntOfTerm(t1),-i)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* two bignums */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_init_set(new, Yap_BigIntOfTerm(t1)); | 
					
						
							|  |  |  | 	mpz_gcd(new, new, Yap_BigIntOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, t2, "gcd/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       mpz_init_set(v1.big,Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       bt1 = big_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt1 = Yap_Eval(t1, &v1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* don't know anything about second */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* second case, no need no evaluation */ | 
					
						
							|  |  |  |   switch (bt1) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = v1.Int, i2 = v2.Int; | 
					
						
							|  |  |  | 	i1 = (i1 >= 0 ? i1 : -i1); | 
					
						
							|  |  |  | 	i2 = (i2 >= 0 ? i2 : -i2); | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  | 	RINT(gcd(i1,i2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v2.dbl), "gcd/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (v1.Int > 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  Int i = mpz_gcd_ui(NULL,v2.big,v1.Int); | 
					
						
							|  |  |  | 	  mpz_clear(v2.big); | 
					
						
							|  |  |  | 	  RINT(i); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} else if (v1.Int == 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  mpz_clear(v2.big); | 
					
						
							|  |  |  | 	  RINT(0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  Int i = mpz_gcd_ui(NULL,v2.big,-v1.Int); | 
					
						
							|  |  |  | 	  mpz_clear(v2.big); | 
					
						
							|  |  |  | 	  RINT(i); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       /* Yap_Error */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case double_e: | 
					
						
							| 
									
										
										
										
											2006-01-02 23:19:10 +00:00
										 |  |  | #if USE_GMP
 | 
					
						
							|  |  |  |     if (bt2 == big_int_e) | 
					
						
							|  |  |  |       mpz_clear(v2.big); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v1.dbl), "gcd/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |     RERROR(); | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* big gcd integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (v2.Int > 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  Int i = mpz_gcd_ui(NULL,v1.big,v2.Int); | 
					
						
							|  |  |  | 	  mpz_clear(v1.big); | 
					
						
							|  |  |  | 	  RINT(i); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} else if (v2.Int == 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  mpz_clear(v1.big); | 
					
						
							|  |  |  | 	  RINT(0); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  Int i = mpz_gcd_ui(NULL,v1.big,-v2.Int); | 
					
						
							|  |  |  | 	  mpz_clear(v1.big); | 
					
						
							|  |  |  | 	  RINT(i); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* big // float */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       Yap_Error(TYPE_ERROR_INTEGER, MkFloatTerm(v2.dbl), "gcd/2"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       /* make GCC happy */ | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       if (v2.Int > 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	mpz_gcd(new, new, v2.big); | 
					
						
							|  |  |  | 	mpz_clear(v2.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RBIG(new); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error  */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* error  */ | 
					
						
							|  |  |  |     RERROR(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   minimum: min(x,y) | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | static E_FUNC | 
					
						
							|  |  |  | p_min(Term t1, Term t2 E_ARGS) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Functor f1 = AritFunctorOfTerm(t1), f2; | 
					
						
							|  |  |  |   blob_type bt1, bt2; | 
					
						
							|  |  |  |   union arith_ret v1, v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (BlobOfFunctor(f1)) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = IntegerOfTerm(t1); | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	RINT((i1 < i2 ? i1 : i2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	/* integer, double */ | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t1); | 
					
						
							|  |  |  | 	Float fl = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	if (i <= fl) { | 
					
						
							|  |  |  | 	  RINT(i); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(fl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	MP_INT *b = Yap_BigIntOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (mpz_cmp_si(b,i) < 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, b); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	RINT(i); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.Int = IntegerOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = long_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	Float fl = FloatOfTerm(t1); | 
					
						
							|  |  |  | 	if (i <= fl) { | 
					
						
							|  |  |  | 	  RINT(i); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(fl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float fl1 = FloatOfTerm(t1); | 
					
						
							|  |  |  | 	Float fl2 = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	if (fl1 <= fl2) { | 
					
						
							|  |  |  | 	  RFLOAT(fl1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(fl2); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float fl1 = FloatOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Float fl2 = mpz_get_d(Yap_BigIntOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	if (fl1 <= fl2) { | 
					
						
							|  |  |  | 	  RFLOAT(fl1); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, Yap_BigIntOfTerm(t2)); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.dbl = FloatOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = double_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	MP_INT *b = Yap_BigIntOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (mpz_cmp_si(b,i) < 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, b); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	RINT(i); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* two bignums */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	MP_INT *b1 = Yap_BigIntOfTerm(t1); | 
					
						
							|  |  |  | 	MP_INT *b2 = Yap_BigIntOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (mpz_cmp(b1,b2) < 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, b1); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, b2); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float fl1 = FloatOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Float fl2 = mpz_get_d(Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	if (fl1 <= fl2) { | 
					
						
							|  |  |  | 	  RFLOAT(fl1); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, Yap_BigIntOfTerm(t1)); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       mpz_init_set(v1.big,Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       bt1 = big_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt1 = Yap_Eval(t1, &v1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* don't know anything about second */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* second case, no need no evaluation */ | 
					
						
							|  |  |  |   switch (bt1) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       RINT((v1.Int < v2.Int ? v1.Int : v2.Int)); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (v1.Int <= v2.dbl) { | 
					
						
							|  |  |  | 	  RINT(v1.Int); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(v2.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (mpz_cmp_si(v2.big,v1.Int) < 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v2.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v2.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RINT(v1.Int); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       /* Yap_Error */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (v2.Int <= v1.dbl) { | 
					
						
							|  |  |  | 	  RINT(v2.Int); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(v1.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* float / float */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (v2.dbl <= v1.dbl) { | 
					
						
							|  |  |  | 	  RFLOAT(v2.dbl); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(v1.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* float / big */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (mpz_get_d(v2.big) <= v1.dbl) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v2.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v2.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RFLOAT(v1.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (mpz_cmp_si(v1.big,v2.Int) < 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v1.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RINT(v2.Int); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* big / float */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (mpz_get_d(v1.big) <= v2.dbl) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v1.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RFLOAT(v2.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* big / big */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	MP_INT *new = TMP_BIG(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	if (mpz_cmp(v1.big,v2.big) < 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	  mpz_clear(v2.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 	  MPZ_SET(new, v2.big); | 
					
						
							|  |  |  | 	  mpz_clear(v1.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error  */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   maximum: max(x,y) | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | static E_FUNC | 
					
						
							|  |  |  | p_max(Term t1, Term t2 E_ARGS) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Functor f1 = AritFunctorOfTerm(t1), f2; | 
					
						
							|  |  |  |   blob_type bt1, bt2; | 
					
						
							|  |  |  |   union arith_ret v1, v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (BlobOfFunctor(f1)) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i1 = IntegerOfTerm(t1); | 
					
						
							|  |  |  | 	Int i2 = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	RINT((i1 > i2 ? i1 : i2)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	/* integer, double */ | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t1); | 
					
						
							|  |  |  | 	Float fl = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	if (i >= fl) { | 
					
						
							|  |  |  | 	  RINT(i); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(fl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	MP_INT *b = Yap_BigIntOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (mpz_cmp_si(b,i) > 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, b); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	RINT(i); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.Int = IntegerOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = long_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t2); | 
					
						
							|  |  |  | 	Float fl = FloatOfTerm(t1); | 
					
						
							|  |  |  | 	if (i >= fl) { | 
					
						
							|  |  |  | 	  RINT(i); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(fl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float fl1 = FloatOfTerm(t1); | 
					
						
							|  |  |  | 	Float fl2 = FloatOfTerm(t2); | 
					
						
							|  |  |  | 	if (fl1 >= fl2) { | 
					
						
							|  |  |  | 	  RFLOAT(fl1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(fl2); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float fl1 = FloatOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Float fl2 = mpz_get_d(Yap_BigIntOfTerm(t2)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	if (fl1 >= fl2) { | 
					
						
							|  |  |  | 	  RFLOAT(fl1); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, Yap_BigIntOfTerm(t2)); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							|  |  |  |       v1.dbl = FloatOfTerm(t1); | 
					
						
							|  |  |  |       bt1 = double_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     f2 = AritFunctorOfTerm(t2); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     switch (BlobOfFunctor(f2)) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Int i = IntegerOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	MP_INT *b = Yap_BigIntOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (mpz_cmp_si(b,i) > 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, b); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	RINT(i); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* two bignums */ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	MP_INT *b1 = Yap_BigIntOfTerm(t1); | 
					
						
							|  |  |  | 	MP_INT *b2 = Yap_BigIntOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (mpz_cmp(b1,b2) > 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, b1); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, b1); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	Float fl1 = FloatOfTerm(t2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | 	Float fl2 = mpz_get_d(Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	if (fl1 >= fl2) { | 
					
						
							|  |  |  | 	  RFLOAT(fl1); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  mpz_init_set(new, Yap_BigIntOfTerm(t1)); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |       mpz_init_set(v1.big,Yap_BigIntOfTerm(t1)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       bt1 = big_int_e; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     /* we've got a full term, need to evaluate it first */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt1 = Yap_Eval(t1, &v1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     /* don't know anything about second */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     bt2 = Yap_Eval(t2, &v2); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   /* second case, no need no evaluation */ | 
					
						
							|  |  |  |   switch (bt1) { | 
					
						
							|  |  |  |   case long_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* two integers */ | 
					
						
							|  |  |  |       RINT((v1.Int > v2.Int ? v1.Int : v2.Int)); | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (v1.Int >= v2.dbl) { | 
					
						
							|  |  |  | 	  RINT(v1.Int); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(v2.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (mpz_cmp_si(v2.big,v1.Int) > 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v2.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v2.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RINT(v1.Int); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       /* Yap_Error */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   case double_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* float / integer */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (v2.Int >= v1.dbl) { | 
					
						
							|  |  |  | 	  RINT(v2.Int); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(v1.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* float / float */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (v2.dbl >= v1.dbl) { | 
					
						
							|  |  |  | 	  RFLOAT(v2.dbl); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RFLOAT(v1.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* float / big */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (mpz_get_d(v2.big) >= v1.dbl) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v2.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v2.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RFLOAT(v1.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #ifdef USE_GMP
 | 
					
						
							|  |  |  |   case big_int_e: | 
					
						
							|  |  |  |     switch (bt2) { | 
					
						
							|  |  |  |     case long_int_e: | 
					
						
							|  |  |  |       /* integer, double */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (mpz_cmp_si(v1.big,v2.Int) > 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v1.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RINT(v2.Int); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case double_e: | 
					
						
							|  |  |  |       /* big / float */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (mpz_get_d(v1.big) >= v2.dbl) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	mpz_clear(v1.big); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	RFLOAT(v2.dbl); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     case big_int_e: | 
					
						
							|  |  |  |       /* big / big */ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	if (mpz_cmp(v1.big,v2.big) > 0) { | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v1.big); | 
					
						
							|  |  |  | 	  mpz_clear(v2.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 	  MP_INT *new = TMP_BIG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  MPZ_SET(new, v2.big); | 
					
						
							|  |  |  | 	  mpz_clear(v1.big); | 
					
						
							|  |  |  | 	  RBIG(new); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* error  */ | 
					
						
							|  |  |  |       RERROR(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static InitBinEntry InitBinTab[] = { | 
					
						
							|  |  |  |   {"+", p_plus}, | 
					
						
							|  |  |  |   {"-", p_minus}, | 
					
						
							|  |  |  |   {"*", p_times}, | 
					
						
							|  |  |  |   {"/", p_fdiv}, | 
					
						
							|  |  |  |   {"mod", p_mod}, | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |   {"rem", p_rem}, | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   {"//", p_div}, | 
					
						
							|  |  |  |   {"<<", p_sll}, | 
					
						
							|  |  |  |   {">>", p_slr}, | 
					
						
							|  |  |  |   {"/\\", p_and}, | 
					
						
							|  |  |  |   {"\\/", p_or}, | 
					
						
							|  |  |  |   {"#", p_xor}, | 
					
						
							|  |  |  |   {"atan2", p_atan2}, | 
					
						
							|  |  |  |   /* C-Prolog exponentiation */ | 
					
						
							|  |  |  |   {"^", p_power}, | 
					
						
							|  |  |  |   /* ISO-Prolog exponentiation */ | 
					
						
							|  |  |  |   {"**", p_power}, | 
					
						
							|  |  |  |   /* Quintus exponentiation */ | 
					
						
							|  |  |  |   {"exp", p_power}, | 
					
						
							|  |  |  |   {"gcd", p_gcd}, | 
					
						
							|  |  |  |   {"min", p_min}, | 
					
						
							| 
									
										
										
										
											2003-04-30 17:46:05 +00:00
										 |  |  |   {"max", p_max} | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int  | 
					
						
							|  |  |  | p_binary_is(void) | 
					
						
							|  |  |  | {				/* X is Y	 */ | 
					
						
							|  |  |  |   Term t = Deref(ARG2); | 
					
						
							|  |  |  |   union arith_ret res; | 
					
						
							|  |  |  |   blob_type f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(t)) { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_Error(INSTANTIATION_ERROR,t, "X is Y"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     return(FALSE); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (IsIntTerm(t)) { | 
					
						
							|  |  |  |     blob_type f = InitBinTab[IntOfTerm(t)].f(Deref(ARG3),Deref(ARG4),&res); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     Term out = EvalToTerm(f,&res); | 
					
						
							|  |  |  |     if (out == TermNil) { | 
					
						
							|  |  |  |       Yap_Error(EVALUATION_ERROR_INT_OVERFLOW, t, "is/2"); | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return (Yap_unify_constant(ARG1,out)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (IsAtomTerm(t)) { | 
					
						
							|  |  |  |     Atom name = AtomOfTerm(t); | 
					
						
							|  |  |  |     ExpEntry *p; | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     Term out; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     if (EndOfPAEntr(p = RepExpProp(Yap_GetExpProp(name, 2)))) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       Term ti[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* error */ | 
					
						
							|  |  |  |       ti[0] = t; | 
					
						
							|  |  |  |       ti[1] = MkIntTerm(2); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |       t = Yap_MkApplTerm(Yap_MkFunctor(Yap_LookupAtom("/"),2), 2, ti); | 
					
						
							|  |  |  |       Yap_Error(TYPE_ERROR_EVALUABLE, t, | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	    "functor %s/%d for arithmetic expression", | 
					
						
							|  |  |  | 	    RepAtom(name)->StrOfAE,2); | 
					
						
							|  |  |  |       P = (yamop *)FAILCODE; | 
					
						
							|  |  |  |       return(FALSE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     f = p->FOfEE.binary(Deref(ARG3),Deref(ARG4),&res); | 
					
						
							| 
									
										
										
										
											2006-01-02 02:16:19 +00:00
										 |  |  |     out = EvalToTerm(f,&res); | 
					
						
							|  |  |  |     if (out == TermNil) { | 
					
						
							|  |  |  |       Yap_Error(EVALUATION_ERROR_INT_OVERFLOW, t, "is/2"); | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return Yap_unify_constant(ARG1,out); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   return(FALSE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_InitBinaryExps(void) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   unsigned int    i; | 
					
						
							|  |  |  |   ExpEntry       *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < sizeof(InitBinTab)/sizeof(InitBinEntry); ++i) { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     AtomEntry *ae = RepAtom(Yap_LookupAtom(InitBinTab[i].OpName)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     WRITE_LOCK(ae->ARWLock); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     if (Yap_GetExpPropHavingLock(ae, 2)) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       WRITE_UNLOCK(ae->ARWLock); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     p = (ExpEntry *) Yap_AllocAtomSpace(sizeof(ExpEntry)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     p->KindOfPE = ExpProperty; | 
					
						
							|  |  |  |     p->ArityOfEE = 2; | 
					
						
							|  |  |  |     p->ENoOfEE = 2; | 
					
						
							|  |  |  |     p->FOfEE.binary = InitBinTab[i].f; | 
					
						
							| 
									
										
										
										
											2001-10-30 16:42:05 +00:00
										 |  |  |     p->NextOfPE = ae->PropsOfAE; | 
					
						
							|  |  |  |     ae->PropsOfAE = AbsExpProp(p); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     WRITE_UNLOCK(ae->ARWLock); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   Yap_InitCPred("is", 4, p_binary_is, TestPredFlag | SafePredFlag); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* This routine is called from Restore to make sure we have the same arithmetic operators */ | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_ReInitBinaryExps(void) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   unsigned int i; | 
					
						
							|  |  |  |   Prop p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < sizeof(InitBinTab)/sizeof(InitBinEntry); ++i) { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     AtomEntry *ae = RepAtom(Yap_FullLookupAtom(InitBinTab[i].OpName)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     WRITE_LOCK(ae->ARWLock); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     if ((p = Yap_GetExpPropHavingLock(ae, 2)) == NULL) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |       WRITE_UNLOCK(ae->ARWLock); | 
					
						
							|  |  |  |       return(FALSE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     RepExpProp(p)->FOfEE.binary = InitBinTab[i].f; | 
					
						
							|  |  |  |     WRITE_UNLOCK(ae->ARWLock); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return(TRUE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 |