diff options
| author | Yukihiro "Matz" Matsumoto <[email protected]> | 2016-05-11 09:41:19 +0900 |
|---|---|---|
| committer | Yukihiro "Matz" Matsumoto <[email protected]> | 2016-05-11 09:41:19 +0900 |
| commit | 5933651ac78b598469130ff95243f2a1674855bc (patch) | |
| tree | 11f3e76a469369d5e3e32f4eb52aba890605cb03 /include | |
| parent | 0be4b8960e1a04ff0681847811bee4398b5664a5 (diff) | |
| parent | 7453a5dfea75e696122ba7895bca76e6ca013d23 (diff) | |
| download | mruby-5933651ac78b598469130ff95243f2a1674855bc.tar.gz mruby-5933651ac78b598469130ff95243f2a1674855bc.zip | |
Merge pull request #3157 from cremno/add-mrb_int_mul_overflow
add function for checked mrb_int multiplication
Diffstat (limited to 'include')
| -rw-r--r-- | include/mruby/numeric.h | 35 |
1 files changed, 35 insertions, 0 deletions
diff --git a/include/mruby/numeric.h b/include/mruby/numeric.h index b16c299a4..e4086487e 100644 --- a/include/mruby/numeric.h +++ b/include/mruby/numeric.h @@ -62,6 +62,12 @@ mrb_int_sub_overflow(mrb_int minuend, mrb_int subtrahend, mrb_int *difference) return __builtin_sub_overflow(minuend, subtrahend, difference) || WBCHK(*difference); } +static inline mrb_bool +mrb_int_mul_overflow(mrb_int multiplier, mrb_int multiplicand, mrb_int *product) +{ + return __builtin_mul_overflow(multiplier, multiplicand, product) || WBCHK(*product); +} + #undef WBCHK #else @@ -92,6 +98,35 @@ mrb_int_sub_overflow(mrb_int minuend, mrb_int subtrahend, mrb_int *difference) return !!(((x ^ z) & (~y ^ z)) & MRB_INT_OVERFLOW_MASK); } +static inline mrb_bool +mrb_int_mul_overflow(mrb_int multiplier, mrb_int multiplicand, mrb_int *product) +{ +#if MRB_INT_BIT == 32 + int64_t n = (int64_t)multiplier * multiplicand; + *product = (mrb_int)n; + return !FIXABLE(n); +#else + if (multiplier > 0) { + if (multiplicand > 0) { + if (multiplier > MRB_INT_MAX / multiplicand) return TRUE; + } + else { + if (multiplicand < MRB_INT_MAX / multiplier) return TRUE; + } + } + else { + if (multiplicand > 0) { + if (multiplier < MRB_INT_MAX / multiplicand) return TRUE; + } + else { + if (multiplier != 0 && multiplicand < MRB_INT_MAX / multiplier) return TRUE; + } + } + *product = multiplier * multiplicand; + return FALSE; +#endif +} + #undef MRB_INT_OVERFLOW_MASK #undef mrb_uint #undef MRB_UINT_MAKE |
