aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorkosaki <kosaki@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2011-11-14 03:54:34 +0000
committerkosaki <kosaki@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2011-11-14 03:54:34 +0000
commitb2a2ba119104b0ff2b6c20611d76de924596ca5b (patch)
tree198f6b99ec7f609a64e199a9efbc6dcb1e542b1e
parentd3437b7c1a496143c2bff2698082af143f5e1dc0 (diff)
downloadruby-b2a2ba119104b0ff2b6c20611d76de924596ca5b.tar.gz
* include/ruby/ruby.h: add NUM2SHORT(), NUM2USHORT() macros.
* numeric.c: ditto. * test/-ext-/num2int/test_num2int.rb: add testcases for NUM2SHORT(). * ext/-test-/num2int/num2int.c: ditto. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@33743 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r--ChangeLog8
-rw-r--r--ext/-test-/num2int/num2int.c25
-rw-r--r--include/ruby/ruby.h19
-rw-r--r--numeric.c74
-rw-r--r--test/-ext-/num2int/test_num2int.rb40
5 files changed, 166 insertions, 0 deletions
diff --git a/ChangeLog b/ChangeLog
index 5c5939531e..c7bb8cea08 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,11 @@
+Sun Nov 13 10:28:18 2011 KOSAKI Motohiro <kosaki.motohiro@gmail.com>
+
+ * include/ruby/ruby.h: add NUM2SHORT(), NUM2USHORT() macros.
+ * numeric.c: ditto.
+
+ * test/-ext-/num2int/test_num2int.rb: add testcases for NUM2SHORT().
+ * ext/-test-/num2int/num2int.c: ditto.
+
Sun Nov 13 10:23:48 2011 KOSAKI Motohiro <kosaki.motohiro@gmail.com>
* bignum.c (rb_big2ull): fix off-by-twice bug of NUM2ULL.
diff --git a/ext/-test-/num2int/num2int.c b/ext/-test-/num2int/num2int.c
index a7242a982c..bafb0e0da9 100644
--- a/ext/-test-/num2int/num2int.c
+++ b/ext/-test-/num2int/num2int.c
@@ -3,6 +3,28 @@
extern VALUE rb_stdout;
static VALUE
+print_num2short(VALUE obj, VALUE num)
+{
+ char buf[128];
+ VALUE str;
+
+ sprintf(buf, "%d", NUM2SHORT(num));
+ str = rb_str_new_cstr(buf);
+ rb_io_write(rb_stdout, str);
+}
+
+static VALUE
+print_num2ushort(VALUE obj, VALUE num)
+{
+ char buf[128];
+ VALUE str;
+
+ sprintf(buf, "%u", NUM2USHORT(num));
+ str = rb_str_new_cstr(buf);
+ rb_io_write(rb_stdout, str);
+}
+
+static VALUE
print_num2int(VALUE obj, VALUE num)
{
char buf[128];
@@ -74,6 +96,9 @@ Init_num2int(void)
{
VALUE cNum2int = rb_path2class("TestNum2int::Num2int");
+ rb_define_singleton_method(cNum2int, "print_num2short", print_num2short, 1);
+ rb_define_singleton_method(cNum2int, "print_num2ushort", print_num2ushort, 1);
+
rb_define_singleton_method(cNum2int, "print_num2int", print_num2int, 1);
rb_define_singleton_method(cNum2int, "print_num2uint", print_num2uint, 1);
diff --git a/include/ruby/ruby.h b/include/ruby/ruby.h
index 03b1a1d1c4..b0dec375d8 100644
--- a/include/ruby/ruby.h
+++ b/include/ruby/ruby.h
@@ -540,6 +540,25 @@ unsigned long rb_fix2uint(VALUE);
#define FIX2UINT(x) ((unsigned int)FIX2ULONG(x))
#endif
+short rb_num2short(VALUE);
+unsigned short rb_num2ushort(VALUE);
+short rb_fix2short(VALUE);
+unsigned short rb_fix2ushort(VALUE);
+#define FIX2SHORT(x) (rb_fix2short((VALUE)(x)))
+#define NUM2SHORT_internal(x) (FIXNUM_P(x) ? FIX2SHORT(x) : rb_num2short(x))
+#ifdef __GNUC__
+# define NUM2SHORT(x) \
+ __extension__ ({VALUE num2short_x = (x); NUM2SHORT_internal(num2short_x);})
+#else /* __GNUC__ */
+static inline short
+NUM2SHORT(VALUE x)
+{
+ return NUM2SHORT_internal(x);
+}
+#endif /* __GNUC__ */
+#define NUM2USHORT(x) rb_num2ushort((VALUE)(x))
+
+
#ifdef HAVE_LONG_LONG
LONG_LONG rb_num2ll(VALUE);
unsigned LONG_LONG rb_num2ull(VALUE);
diff --git a/numeric.c b/numeric.c
index af50fbafde..d3643fd455 100644
--- a/numeric.c
+++ b/numeric.c
@@ -1953,6 +1953,80 @@ rb_fix2int(VALUE val)
}
#endif
+void
+rb_out_of_short(SIGNED_VALUE num)
+{
+ rb_raise(rb_eRangeError, "integer %"PRIdVALUE " too %s to convert to `short'",
+ num, num < 0 ? "small" : "big");
+}
+
+static void
+check_short(SIGNED_VALUE num)
+{
+ if ((SIGNED_VALUE)(short)num != num) {
+ rb_out_of_short(num);
+ }
+}
+
+static void
+check_ushort(VALUE num, VALUE sign)
+{
+ static const VALUE mask = ~(VALUE)USHRT_MAX;
+
+ if (RTEST(sign)) {
+ /* minus */
+ if ((num & mask) != mask || (num & ~mask) <= SHRT_MAX)
+#define VALUE_MSBMASK ((VALUE)1 << ((sizeof(VALUE) * CHAR_BIT) - 1))
+ rb_raise(rb_eRangeError, "integer %"PRIdVALUE " too small to convert to `unsigned short'", num|VALUE_MSBMASK);
+ }
+ else {
+ /* plus */
+ if ((num & mask) != 0)
+ rb_raise(rb_eRangeError, "integer %"PRIuVALUE " too big to convert to `unsigned short'", num);
+ }
+}
+
+short
+rb_num2short(VALUE val)
+{
+ long num = rb_num2long(val);
+
+ check_short(num);
+ return num;
+}
+
+short
+rb_fix2short(VALUE val)
+{
+ long num = FIXNUM_P(val)?FIX2LONG(val):rb_num2long(val);
+
+ check_short(num);
+ return num;
+}
+
+unsigned short
+rb_num2ushort(VALUE val)
+{
+ VALUE num = rb_num2ulong(val);
+
+ check_ushort(num, rb_funcall(val, '<', 1, INT2FIX(0)));
+ return (unsigned long)num;
+}
+
+unsigned short
+rb_fix2ushort(VALUE val)
+{
+ unsigned long num;
+
+ if (!FIXNUM_P(val)) {
+ return rb_num2uint(val);
+ }
+ num = FIX2ULONG(val);
+
+ check_ushort(num, rb_funcall(val, '<', 1, INT2FIX(0)));
+ return num;
+}
+
VALUE
rb_num2fix(VALUE val)
{
diff --git a/test/-ext-/num2int/test_num2int.rb b/test/-ext-/num2int/test_num2int.rb
index b27606288c..6cdfc4c52d 100644
--- a/test/-ext-/num2int/test_num2int.rb
+++ b/test/-ext-/num2int/test_num2int.rb
@@ -5,6 +5,10 @@ class TestNum2int < Test::Unit::TestCase
end
require '-test-/num2int/num2int'
+ SHRT_MIN = -32768
+ SHRT_MAX = 32767
+ USHRT_MAX = 65535
+
INT_MIN = -2147483648
INT_MAX = 2147483647
UINT_MAX = 4294967295
@@ -29,6 +33,42 @@ class TestNum2int < Test::Unit::TestCase
FIXNUM_MAX = LONG_MAX/2
FIXNUM_MIN = LONG_MIN/2
+ def test_num2short
+ assert_output(SHRT_MIN.to_s) do
+ Num2int.print_num2short(SHRT_MIN)
+ end
+ assert_output(SHRT_MAX.to_s) do
+ Num2int.print_num2short(SHRT_MAX)
+ end
+ assert_raise(RangeError) do
+ Num2int.print_num2short(SHRT_MIN-1)
+ end
+ assert_raise(RangeError) do
+ Num2int.print_num2short(SHRT_MAX+1)
+ end
+ end
+
+ def test_num2ushort
+ assert_output("0") do
+ Num2int.print_num2ushort(0)
+ end
+ assert_output(USHRT_MAX.to_s) do
+ Num2int.print_num2ushort(USHRT_MAX)
+ end
+ assert_output(USHRT_MAX.to_s) do
+ Num2int.print_num2ushort(-1)
+ end
+ assert_output((SHRT_MAX+1).to_s) do
+ Num2int.print_num2ushort(SHRT_MIN)
+ end
+ assert_raise(RangeError) do
+ Num2int.print_num2ushort(SHRT_MIN-1)
+ end
+ assert_raise(RangeError) do
+ Num2int.print_num2ushort(USHRT_MAX+1)
+ end
+ end
+
def test_num2int
assert_output(INT_MIN.to_s) do
Num2int.print_num2int(INT_MIN)