summaryrefslogtreecommitdiffhomepage
path: root/include
diff options
context:
space:
mode:
authorYukihiro "Matz" Matsumoto <[email protected]>2016-05-10 10:04:04 +0900
committerYukihiro "Matz" Matsumoto <[email protected]>2016-05-10 10:04:04 +0900
commit4fd0cb61d73ef6e1a78471588ec7e698b2c48041 (patch)
tree198236d866517622fea65483c4b7dd27d4cb07f0 /include
parentbc715f9938eb081c0ee0b1dfb51a249fec32e5e6 (diff)
parent7276e84637f0abb9f4f1e709b13388aff3499663 (diff)
downloadmruby-4fd0cb61d73ef6e1a78471588ec7e698b2c48041.tar.gz
mruby-4fd0cb61d73ef6e1a78471588ec7e698b2c48041.zip
Merge pull request #3154 from cremno/improve-checked-mrb_int-arithmetic-implementation
improve checked mrb int arithmetic implementation
Diffstat (limited to 'include')
-rw-r--r--include/mruby/numeric.h62
1 files changed, 28 insertions, 34 deletions
diff --git a/include/mruby/numeric.h b/include/mruby/numeric.h
index 6a7c172a6..b16c299a4 100644
--- a/include/mruby/numeric.h
+++ b/include/mruby/numeric.h
@@ -31,53 +31,47 @@ mrb_value mrb_fixnum_minus(mrb_state *mrb, mrb_value x, mrb_value y);
mrb_value mrb_fixnum_mul(mrb_state *mrb, mrb_value x, mrb_value y);
mrb_value mrb_num_div(mrb_state *mrb, mrb_value x, mrb_value y);
-#define MRB_UINT_MAKE2(n) uint ## n ## _t
-#define MRB_UINT_MAKE(n) MRB_UINT_MAKE2(n)
-#define mrb_uint MRB_UINT_MAKE(MRB_INT_BIT)
-
-#define MRB_INT_OVERFLOW_MASK ((mrb_uint)1 << (MRB_INT_BIT - 1 - MRB_FIXNUM_SHIFT))
+#ifndef __has_builtin
+ #define __has_builtin(x) 0
+#endif
-/* Idea from Potion: https://github.com/perl11/potion (MIT) */
-#if (defined(__clang__) && ((__clang_major__ > 3) || (__clang_major__ == 3 && __clang_minor__ >= 4))) \
- || (defined(__GNUC__) && __GNUC__ >= 5)
+#if (defined(__GNUC__) && __GNUC__ >= 5) || \
+ (__has_builtin(__builtin_add_overflow) && \
+ __has_builtin(__builtin_sub_overflow) && \
+ __has_builtin(__builtin_mul_overflow))
+# define MRB_HAVE_TYPE_GENERIC_CHECKED_ARITHMETIC_BUILTINS
+#endif
-static inline mrb_bool
-mrb_int_add_overflow(mrb_int augend, mrb_int addend, mrb_int *sum)
-{
- mrb_bool of;
+#ifdef MRB_HAVE_TYPE_GENERIC_CHECKED_ARITHMETIC_BUILTINS
-#ifdef MRB_INT64
- long long val;
- of = __builtin_saddll_overflow(augend, addend, &val) ||
+#ifndef MRB_WORD_BOXING
+# define WBCHK(x) 0
#else
- int val;
- of = __builtin_sadd_overflow(augend, addend, &val) ||
+# define WBCHK(x) !FIXABLE(x)
#endif
- (val > MRB_INT_MAX) || (val < MRB_INT_MIN);
- *sum = (mrb_int) val;
- return of;
+static inline mrb_bool
+mrb_int_add_overflow(mrb_int augend, mrb_int addend, mrb_int *sum)
+{
+ return __builtin_add_overflow(augend, addend, sum) || WBCHK(*sum);
}
static inline mrb_bool
mrb_int_sub_overflow(mrb_int minuend, mrb_int subtrahend, mrb_int *difference)
{
- mrb_bool of;
+ return __builtin_sub_overflow(minuend, subtrahend, difference) || WBCHK(*difference);
+}
-#ifdef MRB_INT64
- long long val;
- of = __builtin_ssubll_overflow(minuend, subtrahend, &val) ||
-#else
- int val;
- of = __builtin_ssub_overflow(minuend, subtrahend, &val) ||
-#endif
- (val > MRB_INT_MAX) || (val < MRB_INT_MIN);
+#undef WBCHK
- *difference = (mrb_int) val;
- return of;
-}
#else
+#define MRB_UINT_MAKE2(n) uint ## n ## _t
+#define MRB_UINT_MAKE(n) MRB_UINT_MAKE2(n)
+#define mrb_uint MRB_UINT_MAKE(MRB_INT_BIT)
+
+#define MRB_INT_OVERFLOW_MASK ((mrb_uint)1 << (MRB_INT_BIT - 1 - MRB_FIXNUM_SHIFT))
+
static inline mrb_bool
mrb_int_add_overflow(mrb_int augend, mrb_int addend, mrb_int *sum)
{
@@ -98,13 +92,13 @@ mrb_int_sub_overflow(mrb_int minuend, mrb_int subtrahend, mrb_int *difference)
return !!(((x ^ z) & (~y ^ z)) & MRB_INT_OVERFLOW_MASK);
}
-#endif
-
#undef MRB_INT_OVERFLOW_MASK
#undef mrb_uint
#undef MRB_UINT_MAKE
#undef MRB_UINT_MAKE2
+#endif
+
MRB_END_DECL
#endif /* MRUBY_NUMERIC_H */