summaryrefslogtreecommitdiff
path: root/numeric.c
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 /numeric.c
parentd3437b7c1a496143c2bff2698082af143f5e1dc0 (diff)
* 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
Diffstat (limited to 'numeric.c')
-rw-r--r--numeric.c74
1 files changed, 74 insertions, 0 deletions
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)
{