Unverified Commit a5b005b5 authored by Yukihiro "Matz" Matsumoto's avatar Yukihiro "Matz" Matsumoto Committed by GitHub

Merge pull request #4493 from dearblue/fix-rational-new

Fix rational new on 32 bits mode
parents 45c99155 64b3d452
#include <mruby.h> #include <mruby.h>
#include <mruby/class.h> #include <mruby/class.h>
#include <mruby/string.h> #include <mruby/string.h>
#include <mruby/istruct.h> #include <mruby/numeric.h>
struct mrb_rational { struct mrb_rational {
mrb_int numerator; mrb_int numerator;
mrb_int denominator; mrb_int denominator;
}; };
#if MRB_INT_MAX <= INTPTR_MAX
#define RATIONAL_USE_ISTRUCT
/* use TT_ISTRUCT */
#include <mruby/istruct.h>
#define rational_ptr(mrb, v) (struct mrb_rational*)mrb_istruct_ptr(v)
static struct RBasic*
rational_alloc(mrb_state *mrb, struct RClass *c, struct mrb_rational **p)
{
struct RIStruct *s;
s = (struct RIStruct*)mrb_obj_alloc(mrb, MRB_TT_ISTRUCT, c);
*p = (struct mrb_rational*)s->inline_data;
return (struct RBasic*)s;
}
#else
/* use TT_DATA */
#include <mruby/data.h>
static const struct mrb_data_type mrb_rational_type = {"Rational", mrb_free};
static struct RBasic*
rational_alloc(mrb_state *mrb, struct RClass *c, struct mrb_rational **p)
{
struct RData *d;
Data_Make_Struct(mrb, c, struct mrb_rational, &mrb_rational_type, *p, d);
return (struct RBasic*)d;
}
static struct mrb_rational* static struct mrb_rational*
rational_ptr(mrb_value v) rational_ptr(mrb_state *mrb, mrb_value v)
{ {
return (struct mrb_rational*)mrb_istruct_ptr(v); struct mrb_rational *p;
p = DATA_GET_PTR(mrb, v, &mrb_rational_type, struct mrb_rational);
if (!p) {
mrb_raise(mrb, E_ARGUMENT_ERROR, "uninitialized rational");
}
return p;
} }
#endif
static mrb_value static mrb_value
rational_numerator(mrb_state *mrb, mrb_value self) rational_numerator(mrb_state *mrb, mrb_value self)
{ {
struct mrb_rational *p = rational_ptr(self); struct mrb_rational *p = rational_ptr(mrb, self);
return mrb_fixnum_value(p->numerator); return mrb_fixnum_value(p->numerator);
} }
static mrb_value static mrb_value
rational_denominator(mrb_state *mrb, mrb_value self) rational_denominator(mrb_state *mrb, mrb_value self)
{ {
struct mrb_rational *p = rational_ptr(self); struct mrb_rational *p = rational_ptr(mrb, self);
return mrb_fixnum_value(p->denominator); return mrb_fixnum_value(p->denominator);
} }
...@@ -32,13 +74,12 @@ static mrb_value ...@@ -32,13 +74,12 @@ static mrb_value
rational_new(mrb_state *mrb, mrb_int numerator, mrb_int denominator) rational_new(mrb_state *mrb, mrb_int numerator, mrb_int denominator)
{ {
struct RClass *c = mrb_class_get(mrb, "Rational"); struct RClass *c = mrb_class_get(mrb, "Rational");
struct RIStruct *s = (struct RIStruct*)mrb_obj_alloc(mrb, MRB_TT_ISTRUCT, c); struct mrb_rational *p;
mrb_value rat = mrb_obj_value(s); struct RBasic *rat = rational_alloc(mrb, c, &p);
struct mrb_rational *p = rational_ptr(rat);
p->numerator = numerator; p->numerator = numerator;
p->denominator = denominator; p->denominator = denominator;
MRB_SET_FROZEN_FLAG(s); MRB_SET_FROZEN_FLAG(rat);
return rat; return mrb_obj_value(rat);
} }
static mrb_value static mrb_value
...@@ -46,7 +87,52 @@ rational_s_new(mrb_state *mrb, mrb_value self) ...@@ -46,7 +87,52 @@ rational_s_new(mrb_state *mrb, mrb_value self)
{ {
mrb_int numerator, denominator; mrb_int numerator, denominator;
#ifdef MRB_WITHOUT_FLOAT
mrb_get_args(mrb, "ii", &numerator, &denominator); mrb_get_args(mrb, "ii", &numerator, &denominator);
#else
#define DROP_PRECISION(cond, num, denom) \
do { \
while (cond) { \
num /= 2; \
denom /= 2; \
} \
} while (0)
mrb_value numv, denomv;
mrb_get_args(mrb, "oo", &numv, &denomv);
if (mrb_fixnum_p(numv)) {
numerator = mrb_fixnum(numv);
if (mrb_fixnum_p(denomv)) {
denominator = mrb_fixnum(denomv);
}
else {
mrb_float denomf = mrb_to_flo(mrb, denomv);
DROP_PRECISION(denomf < MRB_INT_MIN || denomf > MRB_INT_MAX, numerator, denomf);
denominator = denomf;
}
}
else {
mrb_float numf = mrb_to_flo(mrb, numv);
if (mrb_fixnum_p(denomv)) {
denominator = mrb_fixnum(denomv);
}
else {
mrb_float denomf = mrb_to_flo(mrb, denomv);
DROP_PRECISION(denomf < MRB_INT_MIN || denomf > MRB_INT_MAX, numf, denomf);
denominator = denomf;
}
DROP_PRECISION(numf < MRB_INT_MIN || numf > MRB_INT_MAX, numf, denominator);
numerator = numf;
}
#endif
return rational_new(mrb, numerator, denominator); return rational_new(mrb, numerator, denominator);
} }
...@@ -54,7 +140,7 @@ rational_s_new(mrb_state *mrb, mrb_value self) ...@@ -54,7 +140,7 @@ rational_s_new(mrb_state *mrb, mrb_value self)
static mrb_value static mrb_value
rational_to_f(mrb_state *mrb, mrb_value self) rational_to_f(mrb_state *mrb, mrb_value self)
{ {
struct mrb_rational *p = rational_ptr(self); struct mrb_rational *p = rational_ptr(mrb, self);
mrb_float f = (mrb_float)p->numerator / (mrb_float)p->denominator; mrb_float f = (mrb_float)p->numerator / (mrb_float)p->denominator;
return mrb_float_value(mrb, f); return mrb_float_value(mrb, f);
...@@ -64,7 +150,10 @@ rational_to_f(mrb_state *mrb, mrb_value self) ...@@ -64,7 +150,10 @@ rational_to_f(mrb_state *mrb, mrb_value self)
static mrb_value static mrb_value
rational_to_i(mrb_state *mrb, mrb_value self) rational_to_i(mrb_state *mrb, mrb_value self)
{ {
struct mrb_rational *p = rational_ptr(self); struct mrb_rational *p = rational_ptr(mrb, self);
if (p->denominator == 0) {
mrb_raise(mrb, mrb_exc_get(mrb, "StandardError"), "divided by 0");
}
return mrb_fixnum_value(p->numerator / p->denominator); return mrb_fixnum_value(p->numerator / p->denominator);
} }
...@@ -77,7 +166,7 @@ rational_to_r(mrb_state *mrb, mrb_value self) ...@@ -77,7 +166,7 @@ rational_to_r(mrb_state *mrb, mrb_value self)
static mrb_value static mrb_value
rational_negative_p(mrb_state *mrb, mrb_value self) rational_negative_p(mrb_state *mrb, mrb_value self)
{ {
struct mrb_rational *p = rational_ptr(self); struct mrb_rational *p = rational_ptr(mrb, self);
if (p->numerator < 0) { if (p->numerator < 0) {
return mrb_true_value(); return mrb_true_value();
} }
...@@ -94,9 +183,10 @@ void mrb_mruby_rational_gem_init(mrb_state *mrb) ...@@ -94,9 +183,10 @@ void mrb_mruby_rational_gem_init(mrb_state *mrb)
{ {
struct RClass *rat; struct RClass *rat;
#ifdef COMPLEX_USE_RATIONAL
mrb_assert(sizeof(struct mrb_rational) < ISTRUCT_DATA_SIZE); mrb_assert(sizeof(struct mrb_rational) < ISTRUCT_DATA_SIZE);
#endif
rat = mrb_define_class(mrb, "Rational", mrb_class_get(mrb, "Numeric")); rat = mrb_define_class(mrb, "Rational", mrb_class_get(mrb, "Numeric"));
MRB_SET_INSTANCE_TT(rat, MRB_TT_ISTRUCT);
mrb_undef_class_method(mrb, rat, "new"); mrb_undef_class_method(mrb, rat, "new");
mrb_define_class_method(mrb, rat, "_new", rational_s_new, MRB_ARGS_REQ(2)); mrb_define_class_method(mrb, rat, "_new", rational_s_new, MRB_ARGS_REQ(2));
mrb_define_method(mrb, rat, "numerator", rational_numerator, MRB_ARGS_NONE()); mrb_define_method(mrb, rat, "numerator", rational_numerator, MRB_ARGS_NONE());
......
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