Commit e1f7617f authored by Yukihiro Matsumoto's avatar Yukihiro Matsumoto

numeric.c: big restructuring; coercing removed, indentation fixed, unnecessary code removed

parent f373169a
...@@ -28,61 +28,8 @@ ...@@ -28,61 +28,8 @@
#include <ieeefp.h> #include <ieeefp.h>
#endif #endif
/* use IEEE 64bit values if not defined */
#ifndef FLT_RADIX
#define FLT_RADIX 2
#endif
#ifndef FLT_ROUNDS
#define FLT_ROUNDS 1
#endif
#ifndef DBL_MIN
#define DBL_MIN 2.2250738585072014e-308
#endif
#ifndef DBL_MAX
#define DBL_MAX 1.7976931348623157e+308
#endif
#ifndef DBL_MIN_EXP
#define DBL_MIN_EXP (-1021)
#endif
#ifndef DBL_MAX_EXP
#define DBL_MAX_EXP 1024
#endif
#ifndef DBL_MIN_10_EXP
#define DBL_MIN_10_EXP (-307)
#endif
#ifndef DBL_MAX_10_EXP
#define DBL_MAX_10_EXP 308
#endif
#ifndef DBL_DIG
#define DBL_DIG 15
#endif
#ifndef DBL_MANT_DIG
#define DBL_MANT_DIG 53
#endif
#ifndef DBL_EPSILON
#define DBL_EPSILON 2.2204460492503131e-16
#endif
#define mrb_rational_raw1(x) mrb_rational_raw(x, INT2FIX(1))
typedef uintptr_t VALUE;
typedef uintptr_t ID;
#define SIGNED_VALUE intptr_t #define SIGNED_VALUE intptr_t
#ifdef HAVE_INFINITY
#elif BYTE_ORDER == LITTLE_ENDIAN
const unsigned char mrb_infinity[] = "\x00\x00\x80\x7f";
#else
const unsigned char mrb_infinity[] = "\x7f\x80\x00\x00";
#endif
#ifdef HAVE_NAN
#elif BYTE_ORDER == LITTLE_ENDIAN
const unsigned char mrb_nan[] = "\x00\x00\xc0\x7f";
#else
const unsigned char mrb_nan[] = "\x7f\xc0\x00\x00";
#endif
#ifdef MRB_USE_FLOAT #ifdef MRB_USE_FLOAT
#define floor(f) floorf(f) #define floor(f) floorf(f)
#define ceil(f) ceilf(f) #define ceil(f) ceilf(f)
...@@ -90,24 +37,6 @@ const unsigned char mrb_nan[] = "\x7f\xc0\x00\x00"; ...@@ -90,24 +37,6 @@ const unsigned char mrb_nan[] = "\x7f\xc0\x00\x00";
#define fmod(x,y) fmodf(x,y) #define fmod(x,y) fmodf(x,y)
#endif #endif
static mrb_float
mrb_round(mrb_float x)
{
mrb_float f;
if (x > 0.0) {
f = floor(x);
x = f + (x - f >= 0.5);
}
else if (x < 0.0) {
f = ceil(x);
x = f - (f - x >= 0.5);
}
return x;
}
#define round(x) mrb_round(x)
void mrb_cmperr(mrb_state *mrb, mrb_value x, mrb_value y); void mrb_cmperr(mrb_state *mrb, mrb_value x, mrb_value y);
void void
...@@ -116,103 +45,18 @@ mrb_num_zerodiv(mrb_state *mrb) ...@@ -116,103 +45,18 @@ mrb_num_zerodiv(mrb_state *mrb)
mrb_raise(mrb, E_ZERODIVISION_ERROR, "divided by 0"); mrb_raise(mrb, E_ZERODIVISION_ERROR, "divided by 0");
} }
static mrb_float
/* mrb_to_flo(mrb_state *mrb, mrb_value val)
* call-seq: {
* num.coerce(numeric) -> array switch (mrb_type(val)) {
* case MRB_TT_FIXNUM:
* If <i>aNumeric</i> is the same type as <i>num</i>, returns an array return (mrb_float)mrb_fixnum(val);
* containing <i>aNumeric</i> and <i>num</i>. Otherwise, returns an case MRB_TT_FLOAT:
* array with both <i>aNumeric</i> and <i>num</i> represented as break;
* <code>Float</code> objects. This coercion mechanism is used by default:
* Ruby to handle mixed-type numeric operations: it is intended to mrb_raise(mrb, E_TYPE_ERROR, "non float value");
* find a compatible common type between the two operands of the operator. }
* return mrb_float(val);
* 1.coerce(2.5) #=> [2.5, 1.0]
* 1.2.coerce(3) #=> [3.0, 1.2]
* 1.coerce(2) #=> [2, 1]
*/
static mrb_value
num_coerce(mrb_state *mrb, mrb_value x)
{
mrb_value y;
mrb_get_args(mrb, "o", &y);
//if (CLASS_OF(x) == CLASS_OF(y))
if (mrb_class(mrb, x) == mrb_class(mrb, y))
return mrb_assoc_new(mrb, y, x);
x = mrb_Float(mrb, x);
y = mrb_Float(mrb, y);
return mrb_assoc_new(mrb, y, x);
}
static mrb_value
coerce_body(mrb_state *mrb, mrb_value *x)
{
return mrb_funcall(mrb, x[1], "coerce", 1, x[0]);
}
static mrb_value
coerce_rescue(mrb_state *mrb, mrb_value *x)
{
volatile mrb_value v = mrb_inspect(mrb, x[1]);
mrb_raise(mrb, E_TYPE_ERROR, "%s can't be coerced into %s",
mrb_special_const_p(x[1])?
RSTRING_PTR(v):
mrb_obj_classname(mrb, x[1]),
mrb_obj_classname(mrb, x[0]));
return mrb_nil_value(); /* dummy */
}
static int
do_coerce(mrb_state *mrb, mrb_value *x, mrb_value *y, int err)
{
mrb_value ary;
mrb_value a[2];
a[0] = *x; a[1] = *y;
ary = coerce_body(mrb, a);
if (mrb_type(ary) != MRB_TT_ARRAY || RARRAY_LEN(ary) != 2) {
if (err) {
mrb_raise(mrb, E_TYPE_ERROR, "coerce must return [x, y]");
}
return FALSE;
}
*x = RARRAY_PTR(ary)[0];
*y = RARRAY_PTR(ary)[1];
return TRUE;
}
mrb_value
mrb_num_coerce_bin(mrb_state *mrb, mrb_value x, mrb_value y, char* func)
{
do_coerce(mrb, &x, &y, TRUE);
return mrb_funcall(mrb, x, func, 1, y);
}
mrb_value
mrb_num_coerce_cmp(mrb_state *mrb, mrb_value x, mrb_value y, char* func)
{
if (do_coerce(mrb, &x, &y, FALSE))
return mrb_funcall(mrb, x, func, 1, y);
return mrb_nil_value();
}
mrb_value
mrb_num_coerce_relop(mrb_state *mrb, mrb_value x, mrb_value y, char* func)
{
mrb_value c, x0 = x, y0 = y;
if (!do_coerce(mrb, &x, &y, FALSE) ||
mrb_nil_p(c = mrb_funcall(mrb, x, func, 1, y))) {
mrb_cmperr(mrb, x0, y0);
return mrb_nil_value(); /* not reached */
}
return c;
} }
/* /*
...@@ -225,7 +69,7 @@ mrb_num_coerce_relop(mrb_state *mrb, mrb_value x, mrb_value y, char* func) ...@@ -225,7 +69,7 @@ mrb_num_coerce_relop(mrb_state *mrb, mrb_value x, mrb_value y, char* func)
static mrb_value static mrb_value
num_uplus(mrb_state *mrb, mrb_value num) num_uplus(mrb_state *mrb, mrb_value num)
{ {
return num; return num;
} }
/* /*
...@@ -238,27 +82,28 @@ num_uplus(mrb_state *mrb, mrb_value num) ...@@ -238,27 +82,28 @@ num_uplus(mrb_state *mrb, mrb_value num)
static mrb_value static mrb_value
num_uminus(mrb_state *mrb, mrb_value num) num_uminus(mrb_state *mrb, mrb_value num)
{ {
mrb_value zero; return mrb_float_value((mrb_float)0 - mrb_to_flo(mrb, num));
}
zero = mrb_fixnum_value(0);
do_coerce(mrb, &zero, &num, TRUE);
return mrb_funcall(mrb, zero, "-", 1, num); static mrb_value
fix_uminus(mrb_state *mrb, mrb_value num)
{
return mrb_fixnum_value(0 - mrb_fixnum(num));
} }
/* /*
* call-seq: * call-seq:
* num.quo(numeric) -> real * num.quo(numeric) -> real
* *
* Returns most exact division (rational for integers, float for floats). * Returns most exact division.
*/ */
static mrb_value static mrb_value
num_quo(mrb_state *mrb, mrb_value x) num_quo(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
return mrb_funcall(mrb, mrb_float_value((mrb_float)mrb_fixnum(x)), "/", 1, y); return mrb_float_value(mrb_to_flo(mrb, x) / mrb_to_flo(mrb, y));
} }
/* /*
...@@ -276,10 +121,10 @@ num_quo(mrb_state *mrb, mrb_value x) ...@@ -276,10 +121,10 @@ num_quo(mrb_state *mrb, mrb_value x)
static mrb_value static mrb_value
num_abs(mrb_state *mrb, mrb_value num) num_abs(mrb_state *mrb, mrb_value num)
{ {
if (mrb_test(mrb_funcall(mrb, num, "<", 1, mrb_fixnum_value(0)))) { if (mrb_to_flo(mrb, num) < 0) {
return mrb_funcall(mrb, num, "-@", 0); return num_uminus(mrb, num);
} }
return num; return num;
} }
/******************************************************************** /********************************************************************
...@@ -291,17 +136,6 @@ num_abs(mrb_state *mrb, mrb_value num) ...@@ -291,17 +136,6 @@ num_abs(mrb_state *mrb, mrb_value num)
* representation. * representation.
*/ */
mrb_value
mrb_float_new(mrb_float d)
{
//NEWOBJ(flt, struct RFloat);
//OBJSETUP(flt, mrb_cFloat, MRB_TT_FLOAT);
//flt->float_value = d;
//return (mrb_value)flt;
return mrb_float_value(d);
}
/* 15.2.9.3.16(x) */ /* 15.2.9.3.16(x) */
/* /*
* call-seq: * call-seq:
...@@ -316,30 +150,30 @@ mrb_float_new(mrb_float d) ...@@ -316,30 +150,30 @@ mrb_float_new(mrb_float d)
static mrb_value static mrb_value
flo_to_s(mrb_state *mrb, mrb_value flt) flo_to_s(mrb_state *mrb, mrb_value flt)
{ {
char buf[32]; char buf[32];
mrb_float value = mrb_float(flt); mrb_float value = mrb_float(flt);
char *p, *e; char *p, *e;
if (isinf(value)) if (isinf(value))
return mrb_str_new2(mrb, value < 0 ? "-Infinity" : "Infinity"); return mrb_str_new2(mrb, value < 0 ? "-Infinity" : "Infinity");
else if(isnan(value)) else if(isnan(value))
return mrb_str_new2(mrb, "NaN"); return mrb_str_new2(mrb, "NaN");
sprintf(buf, "%#.15g", value); /* ensure to print decimal point */ sprintf(buf, "%#.15g", value); /* ensure to print decimal point */
if (!(e = strchr(buf, 'e'))) {
e = buf + strlen(buf);
}
if (!ISDIGIT(e[-1])) { /* reformat if ended with decimal point (ex 111111111111111.) */
sprintf(buf, "%#.14e", value);
if (!(e = strchr(buf, 'e'))) { if (!(e = strchr(buf, 'e'))) {
e = buf + strlen(buf); e = buf + strlen(buf);
} }
if (!ISDIGIT(e[-1])) { /* reformat if ended with decimal point (ex 111111111111111.) */ }
sprintf(buf, "%#.14e", value); p = e;
if (!(e = strchr(buf, 'e'))) { while (p[-1]=='0' && ISDIGIT(p[-2]))
e = buf + strlen(buf); p--;
} memmove(p, e, strlen(e)+1);
} return mrb_str_new2(mrb, buf);
p = e;
while (p[-1]=='0' && ISDIGIT(p[-2]))
p--;
memmove(p, e, strlen(e)+1);
return mrb_str_new2(mrb, buf);
} }
/* 15.2.9.3.2 */ /* 15.2.9.3.2 */
...@@ -357,15 +191,7 @@ flo_minus(mrb_state *mrb, mrb_value x) ...@@ -357,15 +191,7 @@ flo_minus(mrb_state *mrb, mrb_value x)
mrb_value y; mrb_value y;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
return mrb_float_value(mrb_float(x) - mrb_to_flo(mrb, y));
switch (mrb_type(y)) {
case MRB_TT_FIXNUM:
return mrb_float_value(mrb_float(x) - (mrb_float)mrb_fixnum(y));
case MRB_TT_FLOAT:
return mrb_float_value(mrb_float(x) - mrb_float(y));
default:
return mrb_num_coerce_bin(mrb, x, y, "-");
}
} }
/* 15.2.9.3.3 */ /* 15.2.9.3.3 */
...@@ -383,15 +209,7 @@ flo_mul(mrb_state *mrb, mrb_value x) ...@@ -383,15 +209,7 @@ flo_mul(mrb_state *mrb, mrb_value x)
mrb_value y; mrb_value y;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
return mrb_float_value(mrb_float(x) * mrb_to_flo(mrb, y));
switch (mrb_type(y)) {
case MRB_TT_FIXNUM:
return mrb_float_value(mrb_float(x) * (mrb_float)mrb_fixnum(y));
case MRB_TT_FLOAT:
return mrb_float_value(mrb_float(x) * mrb_float(y));
default:
return mrb_num_coerce_bin(mrb, x, y, "*");
}
} }
/* 15.2.9.3.4 */ /* 15.2.9.3.4 */
...@@ -407,53 +225,28 @@ static mrb_value ...@@ -407,53 +225,28 @@ static mrb_value
flo_div(mrb_state *mrb, mrb_value x) flo_div(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
long f_y;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
return mrb_float_value(mrb_float(x) / mrb_to_flo(mrb, y));
switch (mrb_type(y)) {
case MRB_TT_FIXNUM:
f_y = mrb_fixnum(y);
return mrb_float_value(mrb_float(x) / (mrb_float)f_y);
case MRB_TT_FLOAT:
return mrb_float_value(mrb_float(x) / mrb_float(y));
default:
return mrb_num_coerce_bin(mrb, x, y, "/");
}
}
/*
* call-seq:
* float.quo(numeric) -> float
*
* Returns float / numeric.
*/
static mrb_value
flo_quo(mrb_state *mrb, mrb_value x)
{
mrb_value y;
mrb_get_args(mrb, "o", &y);
return mrb_funcall(mrb, x, "/", 1, y);
} }
static void static void
flodivmod(mrb_state *mrb, mrb_float x, mrb_float y, mrb_float *divp, mrb_float *modp) flodivmod(mrb_state *mrb, mrb_float x, mrb_float y, mrb_float *divp, mrb_float *modp)
{ {
mrb_float div, mod; mrb_float div, mod;
if (y == 0.0) mrb_num_zerodiv(mrb); if (y == 0.0) mrb_num_zerodiv(mrb);
mod = fmod(x, y); mod = fmod(x, y);
if (isinf(x) && !isinf(y) && !isnan(y)) if (isinf(x) && !isinf(y) && !isnan(y))
div = x; div = x;
else else
div = (x - mod) / y; div = (x - mod) / y;
if (y*mod < 0) { if (y*mod < 0) {
mod += y; mod += y;
div -= 1.0; div -= 1.0;
} }
if (modp) *modp = mod; if (modp) *modp = mod;
if (divp) *divp = div; if (divp) *divp = div;
} }
/* 15.2.9.3.5 */ /* 15.2.9.3.5 */
...@@ -475,31 +268,11 @@ flo_mod(mrb_state *mrb, mrb_value x) ...@@ -475,31 +268,11 @@ flo_mod(mrb_state *mrb, mrb_value x)
mrb_float fy, mod; mrb_float fy, mod;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
switch (mrb_type(y)) { fy = mrb_to_flo(mrb, y);
case MRB_TT_FIXNUM: flodivmod(mrb, mrb_float(x), fy, 0, &mod);
fy = (mrb_float)mrb_fixnum(y); return mrb_float_value(mod);
break;
case MRB_TT_FLOAT:
fy = mrb_float(y);
break;
default:
return mrb_num_coerce_bin(mrb, x, y, "%");
}
flodivmod(mrb, mrb_float(x), fy, 0, &mod);
return mrb_float_value(mod);
} }
static mrb_value
flt2ival(mrb_float d)
{
if (FIXABLE(d)) {
d = round(d);
return mrb_fixnum_value((long)d);
}
return mrb_nil_value();
}
/* 15.2.8.3.16 */ /* 15.2.8.3.16 */
/* /*
* call-seq: * call-seq:
...@@ -529,8 +302,8 @@ num_eql(mrb_state *mrb, mrb_value x) ...@@ -529,8 +302,8 @@ num_eql(mrb_state *mrb, mrb_value x)
static mrb_value static mrb_value
num_equal(mrb_state *mrb, mrb_value x, mrb_value y) num_equal(mrb_state *mrb, mrb_value x, mrb_value y)
{ {
if (mrb_obj_equal(mrb, x, y)) return mrb_true_value(); if (mrb_obj_equal(mrb, x, y)) return mrb_true_value();
return mrb_funcall(mrb, y, "==", 1, x); return mrb_funcall(mrb, y, "==", 1, x);
} }
/* 15.2.9.3.7 */ /* 15.2.9.3.7 */
...@@ -553,24 +326,18 @@ flo_eq(mrb_state *mrb, mrb_value x) ...@@ -553,24 +326,18 @@ flo_eq(mrb_state *mrb, mrb_value x)
volatile mrb_float a, b; volatile mrb_float a, b;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
switch (mrb_type(y)) { switch (mrb_type(y)) {
case MRB_TT_FIXNUM: case MRB_TT_FIXNUM:
b = (mrb_float)mrb_fixnum(y); b = (mrb_float)mrb_fixnum(y);
break; break;
case MRB_TT_FLOAT: case MRB_TT_FLOAT:
b = mrb_float(y); b = mrb_float(y);
#if defined(_MSC_VER) && _MSC_VER < 1300 break;
if (isnan(b)) return mrb_false_value(); default:
#endif return num_equal(mrb, x, y);
break; }
default: a = mrb_float(x);
return num_equal(mrb, x, y); return (a == b)?mrb_true_value():mrb_false_value();
}
a = mrb_float(x);
#if defined(_MSC_VER) && _MSC_VER < 1300
if (isnan(a)) return mrb_false_value();
#endif
return (a == b)?mrb_true_value():mrb_false_value();
} }
/* 15.2.8.3.18 */ /* 15.2.8.3.18 */
...@@ -597,16 +364,6 @@ flo_hash(mrb_state *mrb, mrb_value num) ...@@ -597,16 +364,6 @@ flo_hash(mrb_state *mrb, mrb_value num)
return mrb_fixnum_value(hash); return mrb_fixnum_value(hash);
} }
mrb_value
mrb_flt_cmp(double a, double b)
{
if (isnan(a) || isnan(b)) return mrb_nil_value();
if (a == b) return mrb_fixnum_value(0);
if (a > b) return mrb_fixnum_value(1);
if (a < b) return mrb_fixnum_value(-1);
return mrb_nil_value();
}
/* 15.2.9.3.13 */ /* 15.2.9.3.13 */
/* /*
* call-seq: * call-seq:
...@@ -618,7 +375,7 @@ mrb_flt_cmp(double a, double b) ...@@ -618,7 +375,7 @@ mrb_flt_cmp(double a, double b)
static mrb_value static mrb_value
flo_to_f(mrb_state *mrb, mrb_value num) flo_to_f(mrb_state *mrb, mrb_value num)
{ {
return num; return num;
} }
/* 15.2.9.3.11 */ /* 15.2.9.3.11 */
...@@ -637,13 +394,12 @@ flo_to_f(mrb_state *mrb, mrb_value num) ...@@ -637,13 +394,12 @@ flo_to_f(mrb_state *mrb, mrb_value num)
static mrb_value static mrb_value
flo_is_infinite_p(mrb_state *mrb, mrb_value num) flo_is_infinite_p(mrb_state *mrb, mrb_value num)
{ {
mrb_float value = mrb_float(num); mrb_float value = mrb_float(num);
if (isinf(value)) {
return mrb_fixnum_value( value < 0 ? -1 : 1 );
}
return mrb_nil_value(); if (isinf(value)) {
return mrb_fixnum_value( value < 0 ? -1 : 1 );
}
return mrb_nil_value();
} }
/* 15.2.9.3.9 */ /* 15.2.9.3.9 */
...@@ -660,17 +416,11 @@ flo_is_infinite_p(mrb_state *mrb, mrb_value num) ...@@ -660,17 +416,11 @@ flo_is_infinite_p(mrb_state *mrb, mrb_value num)
static mrb_value static mrb_value
flo_is_finite_p(mrb_state *mrb, mrb_value num) flo_is_finite_p(mrb_state *mrb, mrb_value num)
{ {
mrb_float value = mrb_float(num); mrb_float value = mrb_float(num);
#if HAVE_FINITE
if (!finite(value))
return mrb_false_value();
#else
if (isinf(value) || isnan(value))
return mrb_false_value();
#endif
return mrb_true_value(); if (isinf(value) || isnan(value))
return mrb_false_value();
return mrb_true_value();
} }
/* 15.2.9.3.10 */ /* 15.2.9.3.10 */
...@@ -689,14 +439,12 @@ flo_is_finite_p(mrb_state *mrb, mrb_value num) ...@@ -689,14 +439,12 @@ flo_is_finite_p(mrb_state *mrb, mrb_value num)
static mrb_value static mrb_value
flo_floor(mrb_state *mrb, mrb_value num) flo_floor(mrb_state *mrb, mrb_value num)
{ {
mrb_float f = floor(mrb_float(num)); mrb_float f = floor(mrb_float(num));
long val;
if (!FIXABLE(f)) { if (!FIXABLE(f)) {
return mrb_flt2big(mrb, f); return mrb_float_value(f);
} }
val = (long)f; return mrb_fixnum_value((mrb_int)f);
return mrb_fixnum_value(val);
} }
/* 15.2.9.3.8 */ /* 15.2.9.3.8 */
...@@ -716,14 +464,12 @@ flo_floor(mrb_state *mrb, mrb_value num) ...@@ -716,14 +464,12 @@ flo_floor(mrb_state *mrb, mrb_value num)
static mrb_value static mrb_value
flo_ceil(mrb_state *mrb, mrb_value num) flo_ceil(mrb_state *mrb, mrb_value num)
{ {
mrb_float f = ceil(mrb_float(num)); mrb_float f = ceil(mrb_float(num));
long val;
if (!FIXABLE(f)) { if (!FIXABLE(f)) {
return mrb_flt2big(mrb, f); return mrb_float_value(f);
} }
val = (long)f; return mrb_fixnum_value((mrb_int)f);
return mrb_fixnum_value(val);
} }
/* 15.2.9.3.12 */ /* 15.2.9.3.12 */
...@@ -763,40 +509,47 @@ flo_round(mrb_state *mrb, /*int argc, mrb_value *argv,*/ mrb_value num) ...@@ -763,40 +509,47 @@ flo_round(mrb_state *mrb, /*int argc, mrb_value *argv,*/ mrb_value num)
mrb_value nd; mrb_value nd;
mrb_float number, f; mrb_float number, f;
int ndigits = 0, i; int ndigits = 0, i;
long val;
mrb_value *argv; mrb_value *argv;
int argc; int argc;
mrb_get_args(mrb, "*", &argv, &argc); mrb_get_args(mrb, "*", &argv, &argc);
if (argc /*> 0 && mrb_scan_args(argc, argv, "01", &nd) */== 1) { if (argc == 1) {
nd = argv[0]; nd = argv[0];
ndigits = mrb_fixnum(nd); ndigits = mrb_fixnum(nd);
} }
number = mrb_float(num); number = mrb_float(num);
f = 1.0; f = 1.0;
i = abs(ndigits); i = abs(ndigits);
while (--i >= 0) while (--i >= 0)
f = f*10.0; f = f*10.0;
if (isinf(f)) { if (isinf(f)) {
if (ndigits < 0) number = 0; if (ndigits < 0) number = 0;
}
else {
mrb_float d;
if (ndigits < 0) number /= f;
else number *= f;
/* home-made inline implementation of round(3) */
if (number > 0.0) {
d = floor(number);
number = d + (number - d >= 0.5);
} }
else { else if (number < 0.0) {
if (ndigits < 0) number /= f; d = ceil(number);
else number *= f; number = d - (d - number >= 0.5);
number = round(number);
if (ndigits < 0) number *= f;
else number /= f;
} }
if (ndigits > 0) return mrb_float_value(number); number = round(number);
if (ndigits < 0) number *= f;
else number /= f;
}
if (!FIXABLE(number)) { if (ndigits > 0) return mrb_float_value(number);
return mrb_flt2big(mrb, number); return mrb_fixnum_value((mrb_int)number);
}
val = (long)number;
return mrb_fixnum_value(val);
} }
/* 15.2.9.3.14 */ /* 15.2.9.3.14 */
...@@ -813,17 +566,15 @@ flo_round(mrb_state *mrb, /*int argc, mrb_value *argv,*/ mrb_value num) ...@@ -813,17 +566,15 @@ flo_round(mrb_state *mrb, /*int argc, mrb_value *argv,*/ mrb_value num)
static mrb_value static mrb_value
flo_truncate(mrb_state *mrb, mrb_value num) flo_truncate(mrb_state *mrb, mrb_value num)
{ {
mrb_float f = mrb_float(num); mrb_float f = mrb_float(num);
long val;
if (f > 0.0) f = floor(f); if (f > 0.0) f = floor(f);
if (f < 0.0) f = ceil(f); if (f < 0.0) f = ceil(f);
if (!FIXABLE(f)) { if (!FIXABLE(f)) {
return mrb_flt2big(mrb, f); return mrb_float_value(f);
} }
val = (long)f; return mrb_fixnum_value((mrb_int)f);
return mrb_fixnum_value(val);
} }
/* 15.2.8.3.17 */ /* 15.2.8.3.17 */
...@@ -842,7 +593,7 @@ flo_truncate(mrb_state *mrb, mrb_value num) ...@@ -842,7 +593,7 @@ flo_truncate(mrb_state *mrb, mrb_value num)
static mrb_value static mrb_value
num_floor(mrb_state *mrb, mrb_value num) num_floor(mrb_state *mrb, mrb_value num)
{ {
return flo_floor(mrb, mrb_Float(mrb, num)); return flo_floor(mrb, mrb_Float(mrb, num));
} }
/* 15.2.8.3.20 */ /* 15.2.8.3.20 */
...@@ -857,96 +608,9 @@ num_floor(mrb_state *mrb, mrb_value num) ...@@ -857,96 +608,9 @@ num_floor(mrb_state *mrb, mrb_value num)
*/ */
static mrb_value static mrb_value
num_round(mrb_state *mrb, /*int argc, mrb_value* argv,*/ mrb_value num) num_round(mrb_state *mrb, mrb_value num)
{
return flo_round(mrb, /*argc, argv,*/ mrb_Float(mrb, num));
}
SIGNED_VALUE
mrb_num2long(mrb_state *mrb, mrb_value val)
{ {
again: return flo_round(mrb, mrb_Float(mrb, num));
if (mrb_nil_p(val)) {
mrb_raise(mrb, E_TYPE_ERROR, "no implicit conversion from nil to integer");
}
if (FIXNUM_P(val)) return mrb_fixnum(val);
switch (mrb_type(val)) {
case MRB_TT_FLOAT:
if (mrb_float(val) <= (mrb_float)LONG_MAX
&& mrb_float(val) >= (mrb_float)LONG_MIN) {
return (SIGNED_VALUE)(mrb_float(val));
}
else {
char buf[24];
char *s;
snprintf(buf, sizeof(buf), "%-.10g", mrb_float(val));
if ((s = strchr(buf, ' ')) != 0) *s = '\0';
mrb_raise(mrb, E_RANGE_ERROR, "float %s out of range of integer", buf);
}
default:
val = mrb_to_int(mrb, val);
goto again;
}
}
mrb_value
mrb_num2ulong(mrb_state *mrb, mrb_value val)
{
again:
if (mrb_nil_p(val)) {
mrb_raise(mrb, E_TYPE_ERROR, "no implicit conversion from nil to integer");
}
if (FIXNUM_P(val)) return val; /* this is FIX2LONG, inteneded */
switch (mrb_type(val)) {
case MRB_TT_FLOAT:
if (mrb_float(val) <= (mrb_float)LONG_MAX
&& mrb_float(val) >= (mrb_float)LONG_MIN) {
return mrb_fixnum_value(mrb_float(val));
}
else {
char buf[24];
char *s;
snprintf(buf, sizeof(buf), "%-.10g", mrb_float(val));
if ((s = strchr(buf, ' ')) != 0) *s = '\0';
mrb_raise(mrb, E_RANGE_ERROR, "float %s out of range of integer", buf);
}
default:
val = mrb_to_int(mrb, val);
goto again;
}
}
long
mrb_num2int(mrb_state *mrb, mrb_value val)
{
return mrb_num2long(mrb, val);
}
long
mrb_fix2int(mrb_value val)
{
return mrb_fixnum(val);
}
mrb_value
mrb_num2fix(mrb_state *mrb, mrb_value val)
{
long v;
if (FIXNUM_P(val)) return val;
v = mrb_num2long(mrb, val);
if (!FIXABLE(v))
mrb_raise(mrb, E_RANGE_ERROR, "integer %ld out of range of fixnum", v);
return mrb_fixnum_value(v);
} }
/* /*
...@@ -977,7 +641,7 @@ mrb_num2fix(mrb_state *mrb, mrb_value val) ...@@ -977,7 +641,7 @@ mrb_num2fix(mrb_state *mrb, mrb_value val)
static mrb_value static mrb_value
int_to_i(mrb_state *mrb, mrb_value num) int_to_i(mrb_state *mrb, mrb_value num)
{ {
return num; return num;
} }
/* 15.2.8.3.21 */ /* 15.2.8.3.21 */
...@@ -995,8 +659,7 @@ int_to_i(mrb_state *mrb, mrb_value num) ...@@ -995,8 +659,7 @@ int_to_i(mrb_state *mrb, mrb_value num)
static mrb_value static mrb_value
fix_succ(mrb_state *mrb, mrb_value num) fix_succ(mrb_state *mrb, mrb_value num)
{ {
long i = mrb_fixnum(num) + 1; return mrb_fixnum_value(mrb_fixnum(num)+1);
return mrb_fixnum_value(i);
} }
/* 15.2.8.3.19 */ /* 15.2.8.3.19 */
...@@ -1013,43 +676,11 @@ fix_succ(mrb_state *mrb, mrb_value num) ...@@ -1013,43 +676,11 @@ fix_succ(mrb_state *mrb, mrb_value num)
static mrb_value static mrb_value
int_succ(mrb_state *mrb, mrb_value num) int_succ(mrb_state *mrb, mrb_value num)
{ {
if (FIXNUM_P(num)) { if (FIXNUM_P(num)) return fix_succ(mrb, num);
long i = mrb_fixnum(num) + 1; return mrb_funcall(mrb, num, "+", 1, mrb_fixnum_value(1));
return mrb_fixnum_value(i);
}
return mrb_funcall(mrb, num, "+", 1, mrb_fixnum_value(1));
} }
mrb_value #define SQRT_LONG_MAX ((SIGNED_VALUE)1<<((sizeof(mrb_int)*CHAR_BIT-1)/2))
rb_fix2str(mrb_state *mrb, mrb_value x, int base)
{
extern const char ruby_digitmap[];
char buf[sizeof(mrb_int)*CHAR_BIT + 2], *b = buf + sizeof buf;
long val = mrb_fixnum(x);
int neg = 0;
if (base < 2 || 36 < base) {
mrb_raise(mrb, E_ARGUMENT_ERROR, "invalid radix %d", base);
}
if (val == 0) {
return mrb_str_new_cstr(mrb, "0");
}
if (val < 0) {
val = -val;
neg = 1;
}
*--b = '\0';
do {
*--b = ruby_digitmap[(int)(val % base)];
} while (val /= base);
if (neg) {
*--b = '-';
}
return mrb_str_new_cstr(mrb, b);
}
#define SQRT_LONG_MAX ((SIGNED_VALUE)1<<((sizeof(intptr_t)*CHAR_BIT-1)/2))
/*tests if N*N would overflow*/ /*tests if N*N would overflow*/
#define FIT_SQRT_LONG(n) (((n)<SQRT_LONG_MAX)&&((n)>=-SQRT_LONG_MAX)) #define FIT_SQRT_LONG(n) (((n)<SQRT_LONG_MAX)&&((n)>=-SQRT_LONG_MAX))
...@@ -1067,102 +698,54 @@ static mrb_value ...@@ -1067,102 +698,54 @@ static mrb_value
fix_mul(mrb_state *mrb, mrb_value x) fix_mul(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
mrb_int a;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
if (FIXNUM_P(y)) { a = mrb_fixnum(x);
#ifdef __HP_cc if (FIXNUM_P(y)) {
/* avoids an optimization bug of HP aC++/ANSI C B3910B A.06.05 [Jul 25 2005] */ mrb_int b, c;
volatile mrb_value r;
#endif
long a, b;
long c;
mrb_value r;
a = mrb_fixnum(x);
b = mrb_fixnum(y);
if (FIT_SQRT_LONG(a) && FIT_SQRT_LONG(b))
return mrb_fixnum_value(a*b);
c = a * b;
r = mrb_fixnum_value(c);
if (a == 0) return x;
if (mrb_fixnum(r) != c || c/a != b) {
//r = mrb_big_mul(mrb_int2big(a), mrb_int2big(b));
r = mrb_fixnum_value(a*b);
}
return r;
}
switch (mrb_type(y)) {
case MRB_TT_FLOAT:
return mrb_float_value((mrb_float)mrb_fixnum(x) * mrb_float(y));
default:
return mrb_num_coerce_bin(mrb, x, y, "*");
}
}
static void b = mrb_fixnum(y);
fixdivmod(mrb_state *mrb, long x, long y, long *divp, long *modp) if (FIT_SQRT_LONG(a) && FIT_SQRT_LONG(b))
{ return mrb_fixnum_value(a*b);
long div, mod; c = a * b;
r = mrb_fixnum_value(c);
if (y == 0) mrb_num_zerodiv(mrb);
if (y < 0) { if (a == 0) return x;
if (x < 0) if (mrb_fixnum(r) != c || c/a != b) {
div = -x / -y; r = mrb_float_value((mrb_float)a*(mrb_float)b);
else
div = - (x / -y);
}
else {
if (x < 0)
div = - (-x / y);
else
div = x / y;
}
mod = x - div*y;
if ((mod < 0 && y > 0) || (mod > 0 && y < 0)) {
mod += y;
div -= 1;
} }
if (divp) *divp = div; return r;
if (modp) *modp = mod; }
return mrb_float_value((mrb_float)a * mrb_to_flo(mrb, y));
} }
mrb_value rb_big_fdiv(mrb_value x, mrb_value y); static void
fixdivmod(mrb_state *mrb, mrb_int x, mrb_int y, mrb_int *divp, mrb_int *modp)
//mrb_value mrb_rational_reciprocal(mrb_value x);
static mrb_value
fix_divide(mrb_state *mrb, mrb_value x, mrb_value y, char* op)
{ {
if (FIXNUM_P(y)) { mrb_int div, mod;
long div;
fixdivmod(mrb, mrb_fixnum(x), mrb_fixnum(y), &div, 0); if (y == 0) mrb_num_zerodiv(mrb);
return mrb_fixnum_value(div); if (y < 0) {
} if (x < 0)
switch (mrb_type(y)) { div = -x / -y;
case MRB_TT_FLOAT: else
{ div = - (x / -y);
mrb_float div; }
else {
if (*op == '/') { if (x < 0)
div = (mrb_float)mrb_fixnum(x) / mrb_float(y); div = - (-x / y);
return mrb_float_value(div); else
} div = x / y;
else { }
if (mrb_float(y) == 0) mrb_num_zerodiv(mrb); mod = x - div*y;
div = (mrb_float)mrb_fixnum(x) / mrb_float(y); if ((mod < 0 && y > 0) || (mod > 0 && y < 0)) {
return mrb_flt2big(mrb, floor(div)); mod += y;
} div -= 1;
} }
//case MRB_TT_RATIONAL: if (divp) *divp = div;
// if (op == '/' && mrb_fixnum(x) == 1) if (modp) *modp = mod;
// return mrb_rational_reciprocal(y);
/* fall through */
default:
return mrb_num_coerce_bin(mrb, x, y, op);
}
} }
/* 15.2.8.3.4 */ /* 15.2.8.3.4 */
...@@ -1179,9 +762,18 @@ static mrb_value ...@@ -1179,9 +762,18 @@ static mrb_value
fix_div(mrb_state *mrb, mrb_value x) fix_div(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
mrb_int a;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
return fix_divide(mrb, x, y, "/"); a = mrb_fixnum(x);
if (FIXNUM_P(y)) {
mrb_int div;
fixdivmod(mrb, a, mrb_fixnum(y), &div, 0);
return mrb_fixnum_value(div);
}
return mrb_float_value((mrb_float)a / mrb_to_flo(mrb, y));
} }
/* 15.2.8.3.5 */ /* 15.2.8.3.5 */
...@@ -1198,25 +790,23 @@ static mrb_value ...@@ -1198,25 +790,23 @@ static mrb_value
fix_mod(mrb_state *mrb, mrb_value x) fix_mod(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
mrb_int a;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
if (FIXNUM_P(y)) { a = mrb_fixnum(x);
long mod; if (FIXNUM_P(y)) {
mrb_int mod;
fixdivmod(mrb, mrb_fixnum(x), mrb_fixnum(y), 0, &mod); fixdivmod(mrb, a, mrb_fixnum(y), 0, &mod);
return mrb_fixnum_value(mod); return mrb_fixnum_value(mod);
} }
switch (mrb_type(y)) { else {
case MRB_TT_FLOAT: mrb_float mod;
{
mrb_float mod;
flodivmod(mrb, (mrb_float)mrb_fixnum(x), mrb_float(y), 0, &mod); flodivmod(mrb, (mrb_float)a, mrb_to_flo(mrb, y), 0, &mod);
return mrb_float_value(mod); return mrb_float_value(mod);
} }
default:
return mrb_num_coerce_bin(mrb, x, y, "%");
}
} }
/* /*
...@@ -1231,27 +821,21 @@ fix_divmod(mrb_state *mrb, mrb_value x) ...@@ -1231,27 +821,21 @@ fix_divmod(mrb_state *mrb, mrb_value x)
mrb_value y; mrb_value y;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
if (FIXNUM_P(y)) { if (FIXNUM_P(y)) {
long div, mod; mrb_int div, mod;
fixdivmod(mrb, mrb_fixnum(x), mrb_fixnum(y), &div, &mod); fixdivmod(mrb, mrb_fixnum(x), mrb_fixnum(y), &div, &mod);
return mrb_assoc_new(mrb, mrb_fixnum_value(div), mrb_fixnum_value(mod));
}
else {
mrb_float div, mod;
mrb_value a, b;
return mrb_assoc_new(mrb, mrb_fixnum_value(div), mrb_fixnum_value(mod)); flodivmod(mrb, (mrb_float)mrb_fixnum(x), mrb_to_flo(mrb, y), &div, &mod);
} a = mrb_float_value((mrb_int)div);
switch (mrb_type(y)) { b = mrb_float_value(mod);
case MRB_TT_FLOAT: return mrb_assoc_new(mrb, a, b);
{ }
mrb_float div, mod;
volatile mrb_value a, b;
flodivmod(mrb, (mrb_float)mrb_fixnum(x), mrb_float(y), &div, &mod);
a = flt2ival(div);
b = mrb_float_value(mod);
return mrb_assoc_new(mrb, a, b);
}
default:
return mrb_num_coerce_bin(mrb, x, y, "divmod");
}
} }
/* 15.2.8.3.7 */ /* 15.2.8.3.7 */
...@@ -1272,14 +856,16 @@ fix_equal(mrb_state *mrb, mrb_value x) ...@@ -1272,14 +856,16 @@ fix_equal(mrb_state *mrb, mrb_value x)
mrb_value y; mrb_value y;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
if (mrb_obj_equal(mrb, x, y)) return mrb_true_value(); if (mrb_obj_equal(mrb, x, y)) return mrb_true_value();
if (FIXNUM_P(y)) return mrb_false_value(); switch (mrb_type(y)) {
switch (mrb_type(y)) { case MRB_TT_FLOAT:
case MRB_TT_FLOAT: if ((mrb_float)mrb_fixnum(x) == mrb_float(y))
return (mrb_float)mrb_fixnum(x) == mrb_float(y) ? mrb_true_value() : mrb_false_value(); return mrb_true_value();
default: /* fall through */
return num_equal(mrb, x, y); case MRB_TT_FIXNUM:
} default:
return mrb_false_value();
}
} }
/* 15.2.8.3.8 */ /* 15.2.8.3.8 */
...@@ -1296,7 +882,7 @@ fix_equal(mrb_state *mrb, mrb_value x) ...@@ -1296,7 +882,7 @@ fix_equal(mrb_state *mrb, mrb_value x)
static mrb_value static mrb_value
fix_rev(mrb_state *mrb, mrb_value num) fix_rev(mrb_state *mrb, mrb_value num)
{ {
long val = mrb_fixnum(num); mrb_int val = mrb_fixnum(num);
val = ~val; val = ~val;
return mrb_fixnum_value(val); return mrb_fixnum_value(val);
...@@ -1326,15 +912,9 @@ static mrb_value ...@@ -1326,15 +912,9 @@ static mrb_value
fix_and(mrb_state *mrb, mrb_value x) fix_and(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
long val; mrb_int val;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
//if (!FIXNUM_P(y = bit_coerce(mrb, y))) {
// return mrb_big_and(y, x);
//}
if (mrb_type(y) == MRB_TT_FLOAT) {
mrb_raise(mrb, E_TYPE_ERROR, "can't convert Float into Integer");
}
y = bit_coerce(mrb, y); y = bit_coerce(mrb, y);
val = mrb_fixnum(x) & mrb_fixnum(y); val = mrb_fixnum(x) & mrb_fixnum(y);
return mrb_fixnum_value(val); return mrb_fixnum_value(val);
...@@ -1352,15 +932,9 @@ static mrb_value ...@@ -1352,15 +932,9 @@ static mrb_value
fix_or(mrb_state *mrb, mrb_value x) fix_or(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
long val; mrb_int val;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
//if (!FIXNUM_P(y = bit_coerce(mrb, y))) {
// return mrb_big_or(y, x);
//}
if (mrb_type(y) == MRB_TT_FLOAT) {
mrb_raise(mrb, E_TYPE_ERROR, "can't convert Float into Integer");
}
y = bit_coerce(mrb, y); y = bit_coerce(mrb, y);
val = mrb_fixnum(x) | mrb_fixnum(y); val = mrb_fixnum(x) | mrb_fixnum(y);
return mrb_fixnum_value(val); return mrb_fixnum_value(val);
...@@ -1378,22 +952,16 @@ static mrb_value ...@@ -1378,22 +952,16 @@ static mrb_value
fix_xor(mrb_state *mrb, mrb_value x) fix_xor(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
long val; mrb_int val;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
//if (!FIXNUM_P(y = bit_coerce(mrb, y))) {
// return mrb_big_xor(y, x);
//}
if (mrb_type(y) == MRB_TT_FLOAT) {
mrb_raise(mrb, E_TYPE_ERROR, "can't convert Float into Integer");
}
y = bit_coerce(mrb, y); y = bit_coerce(mrb, y);
val = mrb_fixnum(x) ^ mrb_fixnum(y); val = mrb_fixnum(x) ^ mrb_fixnum(y);
return mrb_fixnum_value(val); return mrb_fixnum_value(val);
} }
static mrb_value fix_lshift(mrb_state *mrb, long, unsigned long); static mrb_value fix_lshift(mrb_state *mrb, mrb_int, unsigned long);
static mrb_value fix_rshift(long, unsigned long); static mrb_value fix_rshift(mrb_int, unsigned long);
/* 15.2.8.3.12 */ /* 15.2.8.3.12 */
/* /*
...@@ -1407,15 +975,11 @@ static mrb_value ...@@ -1407,15 +975,11 @@ static mrb_value
mrb_fix_lshift(mrb_state *mrb, mrb_value x) mrb_fix_lshift(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
long val, width; mrb_int val, width;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
val = mrb_fixnum(x); val = mrb_fixnum(x);
//if (!FIXNUM_P(y)) y = bit_coerce(mrb, y);
// return mrb_big_lshift(mrb_int2big(val), y);
if (mrb_type(y) == MRB_TT_FLOAT) {
mrb_raise(mrb, E_TYPE_ERROR, "can't convert Float into Integer");
}
width = mrb_fixnum(y); width = mrb_fixnum(y);
if (width < 0) if (width < 0)
return fix_rshift(val, (unsigned long)-width); return fix_rshift(val, (unsigned long)-width);
...@@ -1423,11 +987,12 @@ mrb_fix_lshift(mrb_state *mrb, mrb_value x) ...@@ -1423,11 +987,12 @@ mrb_fix_lshift(mrb_state *mrb, mrb_value x)
} }
static mrb_value static mrb_value
fix_lshift(mrb_state *mrb, long val, unsigned long width) fix_lshift(mrb_state *mrb, mrb_int val, unsigned long width)
{ {
if (width > (sizeof(intptr_t)*CHAR_BIT-1) if (width > (sizeof(mrb_int)*CHAR_BIT-1)
|| ((unsigned long)abs(val))>>(sizeof(intptr_t)*CHAR_BIT-1-width) > 0) { || ((unsigned long)abs(val))>>(sizeof(mrb_int)*CHAR_BIT-1-width) > 0) {
mrb_raise(mrb, E_RANGE_ERROR, "width(%d) > (sizeof(intptr_t)*CHAR_BIT-1)", width); mrb_raise(mrb, E_RANGE_ERROR, "width(%d) > (%d:sizeof(mrb_int)*CHAR_BIT-1)", width,
sizeof(mrb_int)*CHAR_BIT-1);
} }
val = val << width; val = val << width;
return mrb_fixnum_value(val); return mrb_fixnum_value(val);
...@@ -1445,13 +1010,12 @@ static mrb_value ...@@ -1445,13 +1010,12 @@ static mrb_value
mrb_fix_rshift(mrb_state *mrb, mrb_value x) mrb_fix_rshift(mrb_state *mrb, mrb_value x)
{ {
mrb_value y; mrb_value y;
long i, val; mrb_int i, val;
mrb_get_args(mrb, "o", &y); mrb_get_args(mrb, "o", &y);
val = mrb_fixnum(x); val = mrb_fixnum(x);
//if (!FIXNUM_P(y)) y = bit_coerce(mrb, y);
// return mrb_big_rshift(mrb_int2big(val), y); i = mrb_fixnum(y);
i = mrb_fixnum(y);
if (i == 0) return x; if (i == 0) return x;
if (i < 0) if (i < 0)
return fix_lshift(mrb, val, (unsigned long)-i); return fix_lshift(mrb, val, (unsigned long)-i);
...@@ -1459,9 +1023,9 @@ mrb_fix_rshift(mrb_state *mrb, mrb_value x) ...@@ -1459,9 +1023,9 @@ mrb_fix_rshift(mrb_state *mrb, mrb_value x)
} }
static mrb_value static mrb_value
fix_rshift(long val, unsigned long i) fix_rshift(mrb_int val, unsigned long i)
{ {
if (i >= sizeof(long)*CHAR_BIT-1) { if (i >= sizeof(mrb_int)*CHAR_BIT-1) {
if (val < 0) return mrb_fixnum_value(-1); if (val < 0) return mrb_fixnum_value(-1);
return mrb_fixnum_value(0); return mrb_fixnum_value(0);
} }
...@@ -1523,9 +1087,6 @@ fix_to_f(mrb_state *mrb, mrb_value num) ...@@ -1523,9 +1087,6 @@ fix_to_f(mrb_state *mrb, mrb_value num)
static mrb_int static mrb_int
flt2big(mrb_state *mrb, float d) flt2big(mrb_state *mrb, float d)
{ {
//long i = 0;
//BDIGIT c;
//BDIGIT *digits;
mrb_int z; mrb_int z;
if (isinf(d)) { if (isinf(d)) {
...@@ -1587,41 +1148,6 @@ mrb_fixnum_minus(mrb_state *mrb, mrb_value self) ...@@ -1587,41 +1148,6 @@ mrb_fixnum_minus(mrb_state *mrb, mrb_value self)
return mrb_fixnum_value(x - y); return mrb_fixnum_value(x - y);
} }
/* 15.2.8.3.6 */
/*
* call-seq:
* self.i <=> other.i => -1, 0, +1
* < => -1
* = => 0
* > => +1
* Comparison---Returns -1, 0, or +1 depending on whether <i>fix</i> is
* less than, equal to, or greater than <i>numeric</i>. This is the
* basis for the tests in <code>Comparable</code>.
*/
static mrb_value
mrb_fixnum_cmp(mrb_state *mrb, mrb_value self)
{
mrb_int x, y;
mrb_value vy;
mrb_get_args(mrb, "o", &vy);
if (FIXNUM_P(vy)) {
x = mrb_fixnum(self);
y = mrb_fixnum(vy);
DEBUG(printf("%d <=> %d\n", x, y));
if (x > y)
return mrb_fixnum_value(1);
else if (x < y)
return mrb_fixnum_value(-1);
else
return mrb_fixnum_value(0);
}
else {
return mrb_num_coerce_cmp(mrb, self, vy, "<=>");
}
}
/* 15.2.8.3.29 (x) */ /* 15.2.8.3.29 (x) */
/* /*
* call-seq: * call-seq:
...@@ -1635,7 +1161,7 @@ mrb_value ...@@ -1635,7 +1161,7 @@ mrb_value
mrb_fix2str(mrb_state *mrb, mrb_value x, int base) mrb_fix2str(mrb_state *mrb, mrb_value x, int base)
{ {
char buf[64], *b = buf + sizeof buf; char buf[64], *b = buf + sizeof buf;
long val = mrb_fixnum(x); mrb_int val = mrb_fixnum(x);
int neg = 0; int neg = 0;
if (base < 2 || 36 < base) { if (base < 2 || 36 < base) {
...@@ -1713,21 +1239,24 @@ mrb_fixnum_to_s(mrb_state *mrb, mrb_value self) /* fix_to_s */ ...@@ -1713,21 +1239,24 @@ mrb_fixnum_to_s(mrb_state *mrb, mrb_value self) /* fix_to_s */
* basis for the tests in <code>Comparable</code>. * basis for the tests in <code>Comparable</code>.
*/ */
static mrb_value static mrb_value
mrb_float_cmp(mrb_state *mrb, mrb_value self) num_cmp(mrb_state *mrb, mrb_value self)
{ {
mrb_value vy; mrb_value other;
mrb_float x, y; mrb_float x, y;
x = mrb_float(self); mrb_get_args(mrb, "o", &other);
mrb_get_args(mrb, "o", &vy);
if (FIXNUM_P(vy)) { x = mrb_to_flo(mrb, self);
y = (mrb_float)mrb_fixnum(vy); switch (mrb_type(other)) {
} case MRB_TT_FIXNUM:
else { y = (mrb_float)mrb_fixnum(other);
y = mrb_float(vy); break;
case MRB_TT_FLOAT:
y = mrb_float(other);
break;
default:
return mrb_nil_value();
} }
DEBUG(printf("%f <=> %f\n", x, y));
if (x > y) if (x > y)
return mrb_fixnum_value(1); return mrb_fixnum_value(1);
else { else {
...@@ -1767,8 +1296,8 @@ mrb_init_numeric(mrb_state *mrb) ...@@ -1767,8 +1296,8 @@ mrb_init_numeric(mrb_state *mrb)
mrb_define_method(mrb, numeric, "+@", num_uplus, ARGS_REQ(1)); /* 15.2.7.4.1 */ mrb_define_method(mrb, numeric, "+@", num_uplus, ARGS_REQ(1)); /* 15.2.7.4.1 */
mrb_define_method(mrb, numeric, "-@", num_uminus, ARGS_REQ(1)); /* 15.2.7.4.2 */ mrb_define_method(mrb, numeric, "-@", num_uminus, ARGS_REQ(1)); /* 15.2.7.4.2 */
mrb_define_method(mrb, numeric, "abs", num_abs, ARGS_NONE()); /* 15.2.7.4.3 */ mrb_define_method(mrb, numeric, "abs", num_abs, ARGS_NONE()); /* 15.2.7.4.3 */
mrb_define_method(mrb, numeric, "coerce", num_coerce, ARGS_REQ(1)); /* 15.2.7.4.4 */
mrb_define_method(mrb, numeric, "quo", num_quo, ARGS_REQ(1)); /* 15.2.7.4.5 (x) */ mrb_define_method(mrb, numeric, "quo", num_quo, ARGS_REQ(1)); /* 15.2.7.4.5 (x) */
mrb_define_method(mrb, numeric, "<=>", num_cmp, ARGS_REQ(1)); /* 15.2.9.3.6 */
/* Integer Class */ /* Integer Class */
integer = mrb_define_class(mrb, "Integer", numeric); integer = mrb_define_class(mrb, "Integer", numeric);
...@@ -1776,10 +1305,10 @@ mrb_init_numeric(mrb_state *mrb) ...@@ -1776,10 +1305,10 @@ mrb_init_numeric(mrb_state *mrb)
mrb_define_method(mrb, fixnum, "+", mrb_fixnum_plus, ARGS_REQ(1)); /* 15.2.8.3.1 */ mrb_define_method(mrb, fixnum, "+", mrb_fixnum_plus, ARGS_REQ(1)); /* 15.2.8.3.1 */
mrb_define_method(mrb, fixnum, "-", mrb_fixnum_minus, ARGS_REQ(1)); /* 15.2.8.3.2 */ mrb_define_method(mrb, fixnum, "-", mrb_fixnum_minus, ARGS_REQ(1)); /* 15.2.8.3.2 */
mrb_define_method(mrb, fixnum, "-@", fix_uminus, ARGS_REQ(1)); /* 15.2.7.4.2 */
mrb_define_method(mrb, fixnum, "*", fix_mul, ARGS_REQ(1)); /* 15.2.8.3.3 */ mrb_define_method(mrb, fixnum, "*", fix_mul, ARGS_REQ(1)); /* 15.2.8.3.3 */
mrb_define_method(mrb, fixnum, "/", fix_div, ARGS_REQ(1)); /* 15.2.8.3.4 */ mrb_define_method(mrb, fixnum, "/", fix_div, ARGS_REQ(1)); /* 15.2.8.3.4 */
mrb_define_method(mrb, fixnum, "%", fix_mod, ARGS_REQ(1)); /* 15.2.8.3.5 */ mrb_define_method(mrb, fixnum, "%", fix_mod, ARGS_REQ(1)); /* 15.2.8.3.5 */
mrb_define_method(mrb, fixnum, "<=>", mrb_fixnum_cmp, ARGS_REQ(1)); /* 15.2.8.3.6 */
mrb_define_method(mrb, fixnum, "==", fix_equal, ARGS_REQ(1)); /* 15.2.8.3.7 */ mrb_define_method(mrb, fixnum, "==", fix_equal, ARGS_REQ(1)); /* 15.2.8.3.7 */
mrb_define_method(mrb, fixnum, "~", fix_rev, ARGS_NONE()); /* 15.2.8.3.8 */ mrb_define_method(mrb, fixnum, "~", fix_rev, ARGS_NONE()); /* 15.2.8.3.8 */
mrb_define_method(mrb, fixnum, "&", fix_and, ARGS_REQ(1)); /* 15.2.8.3.9 */ mrb_define_method(mrb, fixnum, "&", fix_and, ARGS_REQ(1)); /* 15.2.8.3.9 */
...@@ -1809,7 +1338,6 @@ mrb_init_numeric(mrb_state *mrb) ...@@ -1809,7 +1338,6 @@ mrb_init_numeric(mrb_state *mrb)
mrb_define_method(mrb, fl, "*", flo_mul, ARGS_REQ(1)); /* 15.2.9.3.3 */ mrb_define_method(mrb, fl, "*", flo_mul, ARGS_REQ(1)); /* 15.2.9.3.3 */
mrb_define_method(mrb, fl, "/", flo_div, ARGS_REQ(1)); /* 15.2.9.3.4 */ mrb_define_method(mrb, fl, "/", flo_div, ARGS_REQ(1)); /* 15.2.9.3.4 */
mrb_define_method(mrb, fl, "%", flo_mod, ARGS_REQ(1)); /* 15.2.9.3.5 */ mrb_define_method(mrb, fl, "%", flo_mod, ARGS_REQ(1)); /* 15.2.9.3.5 */
mrb_define_method(mrb, fl, "<=>", mrb_float_cmp, ARGS_REQ(1)); /* 15.2.9.3.6 */
mrb_define_method(mrb, fl, "==", flo_eq, ARGS_REQ(1)); /* 15.2.9.3.7 */ mrb_define_method(mrb, fl, "==", flo_eq, ARGS_REQ(1)); /* 15.2.9.3.7 */
mrb_define_method(mrb, fl, "ceil", flo_ceil, ARGS_NONE()); /* 15.2.9.3.8 */ mrb_define_method(mrb, fl, "ceil", flo_ceil, ARGS_NONE()); /* 15.2.9.3.8 */
mrb_define_method(mrb, fl, "finite?", flo_is_finite_p, ARGS_NONE()); /* 15.2.9.3.9 */ mrb_define_method(mrb, fl, "finite?", flo_is_finite_p, ARGS_NONE()); /* 15.2.9.3.9 */
...@@ -1823,5 +1351,5 @@ mrb_init_numeric(mrb_state *mrb) ...@@ -1823,5 +1351,5 @@ mrb_init_numeric(mrb_state *mrb)
mrb_define_method(mrb, fl, "to_s", flo_to_s, ARGS_NONE()); /* 15.2.9.3.16(x) */ mrb_define_method(mrb, fl, "to_s", flo_to_s, ARGS_NONE()); /* 15.2.9.3.16(x) */
//mrb_define_method(mrb, fl, "<", flo_lt, ARGS_REQ(1)); /* 15.2.9.3.17(x) */ //mrb_define_method(mrb, fl, "<", flo_lt, ARGS_REQ(1)); /* 15.2.9.3.17(x) */
//mrb_define_method(mrb, fl, ">", flo_gt, ARGS_REQ(1)); /* 15.2.9.3.18(x) */ //mrb_define_method(mrb, fl, ">", flo_gt, ARGS_REQ(1)); /* 15.2.9.3.18(x) */
mrb_define_method(mrb, fl, "quo", flo_quo, ARGS_REQ(1)); /* 15.2.9.3.19(x) */ mrb_define_method(mrb, fl, "quo", flo_div, ARGS_REQ(1)); /* 15.2.9.3.19(x) */
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment