2001-04-09 20:54:03 +01:00
|
|
|
/*************************************************************************
|
|
|
|
* *
|
|
|
|
* YAP Prolog @(#)eval.h 1.2
|
|
|
|
* *
|
|
|
|
* Yap Prolog was developed at NCCUP - Universidade do Porto *
|
|
|
|
* *
|
|
|
|
* Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997 *
|
|
|
|
* *
|
|
|
|
**************************************************************************
|
|
|
|
* *
|
|
|
|
* File: eval.h *
|
|
|
|
* Last rev: *
|
|
|
|
* mods: *
|
|
|
|
* comments: arithmetical functions info *
|
|
|
|
* *
|
|
|
|
*************************************************************************/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
/* C library used to implement floating point functions */
|
2002-06-25 17:13:56 +01:00
|
|
|
#if HAVE_MATH_H
|
2001-04-09 20:54:03 +01:00
|
|
|
#include <math.h>
|
2002-06-25 17:13:56 +01:00
|
|
|
#endif
|
2009-10-20 09:21:59 +01:00
|
|
|
#ifdef HAVE_FLOAT_H
|
|
|
|
#include <float.h>
|
|
|
|
#endif
|
2008-03-25 22:03:14 +00:00
|
|
|
#ifdef HAVE_IEEEFP_H
|
2001-04-09 20:54:03 +01:00
|
|
|
#include <ieeefp.h>
|
|
|
|
#endif
|
2008-03-25 22:03:14 +00:00
|
|
|
#ifdef HAVE_LIMITS_H
|
2001-04-09 20:54:03 +01:00
|
|
|
#include <limits.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef LONG_MAX
|
|
|
|
#define Int_MAX LONG_MAX
|
|
|
|
#else
|
|
|
|
#define Int_MAX ((Int)((~((CELL)0))>>1))
|
|
|
|
#endif
|
|
|
|
#ifdef LONG_MIN
|
|
|
|
#define Int_MIN LONG_MIN
|
|
|
|
#else
|
2001-10-30 16:42:05 +00:00
|
|
|
#define Int_MIN (-Int_MAX-(CELL)1)
|
2001-04-09 20:54:03 +01:00
|
|
|
#endif
|
|
|
|
|
2008-12-04 23:33:32 +00:00
|
|
|
typedef enum {
|
|
|
|
op_pi,
|
|
|
|
op_e,
|
2009-10-20 09:21:59 +01:00
|
|
|
op_epsilon,
|
2008-12-04 23:33:32 +00:00
|
|
|
op_inf,
|
|
|
|
op_nan,
|
|
|
|
op_random,
|
|
|
|
op_cputime,
|
|
|
|
op_heapused,
|
|
|
|
op_localsp,
|
|
|
|
op_globalsp,
|
|
|
|
op_b,
|
|
|
|
op_env,
|
|
|
|
op_tr,
|
|
|
|
op_stackfree
|
|
|
|
} arith0_op;
|
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
op_uplus,
|
|
|
|
op_uminus,
|
|
|
|
op_unot,
|
|
|
|
op_exp,
|
|
|
|
op_log,
|
|
|
|
op_log10,
|
|
|
|
op_sqrt,
|
|
|
|
op_sin,
|
|
|
|
op_cos,
|
|
|
|
op_tan,
|
|
|
|
op_sinh,
|
|
|
|
op_cosh,
|
|
|
|
op_tanh,
|
|
|
|
op_asin,
|
|
|
|
op_acos,
|
|
|
|
op_atan,
|
|
|
|
op_asinh,
|
|
|
|
op_acosh,
|
|
|
|
op_atanh,
|
|
|
|
op_floor,
|
|
|
|
op_ceiling,
|
|
|
|
op_round,
|
|
|
|
op_truncate,
|
|
|
|
op_integer,
|
|
|
|
op_float,
|
|
|
|
op_abs,
|
2009-10-20 09:50:51 +01:00
|
|
|
op_lsb,
|
2008-12-04 23:33:32 +00:00
|
|
|
op_msb,
|
2009-10-20 10:03:10 +01:00
|
|
|
op_popcount,
|
2008-12-04 23:33:32 +00:00
|
|
|
op_ffracp,
|
|
|
|
op_fintp,
|
|
|
|
op_sign,
|
|
|
|
op_lgamma,
|
2010-03-05 00:01:04 +00:00
|
|
|
op_erf,
|
|
|
|
op_erfc,
|
2010-05-28 09:53:56 +01:00
|
|
|
op_rational,
|
|
|
|
op_rationalize,
|
2008-12-04 23:33:32 +00:00
|
|
|
op_random1
|
|
|
|
} arith1_op;
|
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
op_plus,
|
|
|
|
op_minus,
|
|
|
|
op_times,
|
|
|
|
op_fdiv,
|
|
|
|
op_mod,
|
|
|
|
op_rem,
|
|
|
|
op_div,
|
2010-08-31 03:50:33 +01:00
|
|
|
op_idiv,
|
2008-12-04 23:33:32 +00:00
|
|
|
op_sll,
|
|
|
|
op_slr,
|
|
|
|
op_and,
|
|
|
|
op_or,
|
|
|
|
op_xor,
|
|
|
|
op_atan2,
|
|
|
|
/* C-Prolog exponentiation */
|
|
|
|
op_power,
|
|
|
|
/* ISO-Prolog exponentiation */
|
|
|
|
/* op_power, */
|
2009-04-25 01:03:00 +01:00
|
|
|
op_power2,
|
2008-12-04 23:33:32 +00:00
|
|
|
/* Quintus exponentiation */
|
|
|
|
/* op_power, */
|
|
|
|
op_gcd,
|
|
|
|
op_min,
|
2010-05-27 12:24:15 +01:00
|
|
|
op_max,
|
|
|
|
op_rdiv
|
2008-12-04 23:33:32 +00:00
|
|
|
} arith2_op;
|
|
|
|
|
|
|
|
Functor STD_PROTO(EvalArg,(Term));
|
2001-04-09 20:54:03 +01:00
|
|
|
|
|
|
|
/* Needed to handle numbers:
|
|
|
|
these two macros are fundamental in the integer/float conversions */
|
|
|
|
|
|
|
|
#ifdef C_PROLOG
|
|
|
|
#define FlIsInt(X) ( (X) == (Int)(X) && IntInBnd((X)) )
|
|
|
|
#else
|
|
|
|
#define FlIsInt(X) ( FALSE )
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef M_WILLIAMS
|
|
|
|
#define MkEvalFl(X) MkFloatTerm(X)
|
|
|
|
#else
|
|
|
|
#define MkEvalFl(X) ( FlIsInt(X) ? MkIntTerm((Int)(X)) : MkFloatTerm(X) )
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/* Macros used by some of the eval functions */
|
|
|
|
#define REvalInt(I) { eval_int = (I); return(FInt); }
|
|
|
|
#define REvalFl(F) { eval_flt = (F); return(FFloat); }
|
|
|
|
#define REvalError() { return(FError); }
|
|
|
|
|
|
|
|
/* this macro, dependent on the particular implementation
|
|
|
|
is used to interface the arguments into the C libraries */
|
2008-03-25 22:03:14 +00:00
|
|
|
#ifdef MPW
|
2001-04-09 20:54:03 +01:00
|
|
|
#define FL(X) ((extended)(X))
|
|
|
|
#else
|
|
|
|
#define FL(X) ((double)(X))
|
|
|
|
#endif
|
|
|
|
|
2002-11-18 18:18:05 +00:00
|
|
|
void STD_PROTO(Yap_InitConstExps,(void));
|
|
|
|
void STD_PROTO(Yap_InitUnaryExps,(void));
|
|
|
|
void STD_PROTO(Yap_InitBinaryExps,(void));
|
2001-04-09 20:54:03 +01:00
|
|
|
|
2002-11-18 18:18:05 +00:00
|
|
|
int STD_PROTO(Yap_ReInitConstExps,(void));
|
|
|
|
int STD_PROTO(Yap_ReInitUnaryExps,(void));
|
|
|
|
int STD_PROTO(Yap_ReInitBinaryExps,(void));
|
2001-04-09 20:54:03 +01:00
|
|
|
|
2008-12-04 23:33:32 +00:00
|
|
|
Term STD_PROTO(Yap_eval_atom,(Int));
|
|
|
|
Term STD_PROTO(Yap_eval_unary,(Int,Term));
|
|
|
|
Term STD_PROTO(Yap_eval_binary,(Int,Term,Term));
|
|
|
|
|
2010-01-04 23:16:42 +00:00
|
|
|
Term STD_PROTO(Yap_InnerEval,(Term));
|
2009-05-22 17:21:39 +01:00
|
|
|
Int STD_PROTO(Yap_ArithError,(yap_error_number,Term,char *msg, ...));
|
|
|
|
|
2012-06-29 22:44:08 +01:00
|
|
|
#include "inline-only.h"
|
2012-06-30 19:42:14 +01:00
|
|
|
INLINE_ONLY inline EXTERN Term
|
|
|
|
Yap_Eval(Term t);
|
2012-06-29 22:44:08 +01:00
|
|
|
|
2012-06-30 19:42:14 +01:00
|
|
|
INLINE_ONLY inline EXTERN Term
|
2010-01-04 23:16:42 +00:00
|
|
|
Yap_Eval(Term t)
|
|
|
|
{
|
2010-01-11 10:35:36 +00:00
|
|
|
if (t == 0L || ( !IsVarTerm(t) && IsNumTerm(t) ))
|
2010-01-04 23:16:42 +00:00
|
|
|
return t;
|
|
|
|
return Yap_InnerEval(t);
|
|
|
|
}
|
|
|
|
|
2009-05-22 17:21:39 +01:00
|
|
|
inline static Term
|
|
|
|
Yap_FoundArithError(Term t, Term inp)
|
2011-03-07 16:02:55 +00:00
|
|
|
{
|
|
|
|
CACHE_REGS
|
2011-05-23 16:19:47 +01:00
|
|
|
if (LOCAL_Error_TYPE) {
|
|
|
|
Yap_Error(LOCAL_Error_TYPE, (inp ? inp : LOCAL_Error_Term), LOCAL_ErrorMessage);
|
2010-01-04 23:16:42 +00:00
|
|
|
P = FAILCODE;
|
|
|
|
return 0L;
|
2009-05-22 17:21:39 +01:00
|
|
|
}
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2008-12-04 23:33:32 +00:00
|
|
|
|
|
|
|
#define RINT(v) return(MkIntegerTerm(v))
|
|
|
|
#define RFLOAT(v) return(MkFloatTerm(v))
|
|
|
|
#define RBIG(v) return(Yap_MkBigIntTerm(v))
|
|
|
|
#define RERROR() return(0L)
|
|
|
|
|
|
|
|
static inline blob_type
|
|
|
|
ETypeOfTerm(Term t)
|
|
|
|
{
|
|
|
|
if (IsIntTerm(t))
|
|
|
|
return long_int_e;
|
|
|
|
if (IsApplTerm(t)) {
|
|
|
|
Functor f = FunctorOfTerm(t);
|
|
|
|
if (f == FunctorDouble)
|
|
|
|
return double_e;
|
|
|
|
if (f == FunctorLongInt)
|
|
|
|
return long_int_e;
|
2010-05-27 12:24:15 +01:00
|
|
|
if (f == FunctorBigInt) {
|
2008-12-04 23:33:32 +00:00
|
|
|
return big_int_e;
|
2010-05-27 12:24:15 +01:00
|
|
|
}
|
2008-12-04 23:33:32 +00:00
|
|
|
}
|
|
|
|
return db_ref_e;
|
|
|
|
}
|
2008-11-28 15:54:08 +00:00
|
|
|
|
|
|
|
#if USE_GMP
|
2010-05-27 12:24:15 +01:00
|
|
|
Term STD_PROTO(Yap_gmq_rdiv_int_int,(Int, Int));
|
|
|
|
Term STD_PROTO(Yap_gmq_rdiv_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmq_rdiv_big_int,(Term, Int));
|
|
|
|
Term STD_PROTO(Yap_gmq_rdiv_big_big,(Term, Term));
|
|
|
|
|
2008-12-04 23:33:32 +00:00
|
|
|
Term STD_PROTO(Yap_gmp_add_ints,(Int, Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_sub_ints,(Int, Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_mul_ints,(Int, Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_sll_ints,(Int, Int));
|
2010-05-27 12:24:15 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_add_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_sub_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_sub_big_int,(Term, Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_mul_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_div_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_div_big_int,(Term, Int));
|
2010-08-31 03:50:33 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_div2_big_int,(Term, Int));
|
2010-05-27 12:24:15 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_fdiv_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_fdiv_big_int,(Term, Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_and_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_ior_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_xor_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_sll_big_int,(Term, Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_add_big_big,(Term, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_sub_big_big,(Term, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_mul_big_big,(Term, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_div_big_big,(Term, Term));
|
2010-08-31 03:50:33 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_div2_big_big,(Term, Term));
|
2010-05-27 12:24:15 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_fdiv_big_big,(Term, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_and_big_big,(Term, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_ior_big_big,(Term, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_xor_big_big,(Term, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_mod_big_big,(Term, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_mod_big_int,(Term, Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_mod_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_rem_big_big,(Term, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_rem_big_int,(Term, Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_rem_int_big,(Int, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_exp_int_int,(Int,Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_exp_int_big,(Int,Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_exp_big_int,(Term,Int));
|
|
|
|
Term STD_PROTO(Yap_gmp_exp_big_big,(Term,Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_gcd_int_big,(Int,Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_gcd_big_big,(Term,Term));
|
2008-11-28 15:54:08 +00:00
|
|
|
|
2009-06-05 18:18:10 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_big_from_64bits,(YAP_LONG_LONG));
|
2008-11-28 15:54:08 +00:00
|
|
|
|
2010-05-28 09:53:56 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_float_to_big,(Float));
|
|
|
|
Term STD_PROTO(Yap_gmp_float_to_rational,(Float));
|
|
|
|
Term STD_PROTO(Yap_gmp_float_rationalize,(Float));
|
2010-05-27 12:24:15 +01:00
|
|
|
Float STD_PROTO(Yap_gmp_to_float,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_add_float_big,(Float, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_sub_float_big,(Float, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_sub_big_float,(Term, Float));
|
|
|
|
Term STD_PROTO(Yap_gmp_mul_float_big,(Float, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_fdiv_float_big,(Float, Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_fdiv_big_float,(Term, Float));
|
|
|
|
|
|
|
|
int STD_PROTO(Yap_gmp_cmp_big_int,(Term, Int));
|
|
|
|
#define Yap_gmp_cmp_int_big(I, T) (-Yap_gmp_cmp_big_int(T, I))
|
|
|
|
int STD_PROTO(Yap_gmp_cmp_big_float,(Term, Float));
|
|
|
|
#define Yap_gmp_cmp_float_big(D, T) (-Yap_gmp_cmp_big_float(T, D))
|
|
|
|
int STD_PROTO(Yap_gmp_cmp_big_big,(Term, Term));
|
|
|
|
|
2010-05-28 12:07:01 +01:00
|
|
|
int STD_PROTO(Yap_gmp_tcmp_big_int,(Term, Int));
|
|
|
|
#define Yap_gmp_tcmp_int_big(I, T) (-Yap_gmp_tcmp_big_int(T, I))
|
|
|
|
int STD_PROTO(Yap_gmp_tcmp_big_float,(Term, Float));
|
|
|
|
#define Yap_gmp_tcmp_float_big(D, T) (-Yap_gmp_tcmp_big_float(T, D))
|
|
|
|
int STD_PROTO(Yap_gmp_tcmp_big_big,(Term, Term));
|
|
|
|
|
2010-05-27 12:24:15 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_neg_int,(Int));
|
2010-05-28 09:53:56 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_abs_big,(Term));
|
2010-05-27 12:24:15 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_neg_big,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_unot_big,(Term));
|
2010-05-28 09:53:56 +01:00
|
|
|
Term STD_PROTO(Yap_gmp_floor,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_ceiling,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_round,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_trunc,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_float_fractional_part,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_float_integer_part,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_sign,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_lsb,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_msb,(Term));
|
|
|
|
Term STD_PROTO(Yap_gmp_popcount,(Term));
|
2010-05-27 12:24:15 +01:00
|
|
|
|
2010-05-28 15:29:20 +01:00
|
|
|
char * STD_PROTO(Yap_gmp_to_string,(Term, char *, size_t, int));
|
|
|
|
size_t STD_PROTO(Yap_gmp_to_size,(Term, int));
|
2010-06-18 23:24:36 +01:00
|
|
|
|
|
|
|
int STD_PROTO(Yap_term_to_existing_big,(Term, MP_INT *));
|
|
|
|
int STD_PROTO(Yap_term_to_existing_rat,(Term, MP_RAT *));
|
2013-03-26 20:01:52 +00:00
|
|
|
|
|
|
|
void Yap_gmp_set_bit(Int i, Term t);
|
2008-11-28 15:54:08 +00:00
|
|
|
#endif
|
2009-06-05 18:18:10 +01:00
|
|
|
|
2013-03-26 20:01:52 +00:00
|
|
|
#define Yap_Mk64IntegerTerm(i) __Yap_Mk64IntegerTerm((i) PASS_REGS)
|
|
|
|
|
|
|
|
INLINE_ONLY inline EXTERN Term __Yap_Mk64IntegerTerm(YAP_LONG_LONG USES_REGS);
|
2009-06-05 18:18:10 +01:00
|
|
|
|
2012-06-30 19:42:14 +01:00
|
|
|
INLINE_ONLY inline EXTERN Term
|
2013-03-26 20:01:52 +00:00
|
|
|
__Yap_Mk64IntegerTerm(YAP_LONG_LONG i USES_REGS)
|
2009-06-05 18:18:10 +01:00
|
|
|
{
|
2009-06-05 20:19:02 +01:00
|
|
|
if (i <= Int_MAX && i >= Int_MIN) {
|
2009-06-05 18:18:10 +01:00
|
|
|
return MkIntegerTerm((Int)i);
|
|
|
|
} else {
|
|
|
|
#if USE_GMP
|
|
|
|
return Yap_gmp_big_from_64bits(i);
|
|
|
|
#else
|
|
|
|
return MkIntTerm(-1);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|