summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorYukihiro "Matz" Matsumoto <[email protected]>2021-01-29 23:51:26 +0900
committerGitHub <[email protected]>2021-01-29 23:51:26 +0900
commitdd603d494dcc2393183dc9a684cb1c3897b45721 (patch)
tree5aca16026c30dbef3a7693bcca300fe242c0811b
parent825241205f07d387ea2490e43de58658cbc35ab6 (diff)
parentb500e8295001849cf96e88d7f0a86d517299ff7d (diff)
downloadmruby-dd603d494dcc2393183dc9a684cb1c3897b45721.tar.gz
mruby-dd603d494dcc2393183dc9a684cb1c3897b45721.zip
Merge pull request #5306 from dearblue/unpack
Fixed `String#unpack` to handle the highest range of integer values
-rw-r--r--mrbgems/mruby-pack/src/pack.c46
1 files changed, 13 insertions, 33 deletions
diff --git a/mrbgems/mruby-pack/src/pack.c b/mrbgems/mruby-pack/src/pack.c
index 014679a5d..b0da7b06b 100644
--- a/mrbgems/mruby-pack/src/pack.c
+++ b/mrbgems/mruby-pack/src/pack.c
@@ -15,6 +15,9 @@
#include <errno.h>
#include <string.h>
+#define INT_OVERFLOW_P(n) ((n) < MRB_INT_MIN || (n) > MRB_INT_MAX)
+#define UINT_OVERFLOW_P(n) ((n) > MRB_INT_MAX)
+
struct tmpl {
mrb_value str;
int idx;
@@ -213,26 +216,6 @@ u32tostr(char *buf, size_t len, uint32_t n)
snprintf(buf, len, "%" PRIu32, n);
#endif /* MRB_NO_STDIO */
}
-
-static void
-i32tostr(char *buf, size_t len, int32_t n)
-{
-#ifdef MRB_NO_STDIO
- if (len < 1) {
- return;
- }
-
- if (n < 0) {
- *buf ++ = '-';
- len --;
- n = -n;
- }
-
- u32tostr(buf, len, (uint32_t)n);
-#else
- snprintf(buf, len, "%" PRId32, n);
-#endif /* MRB_NO_STDIO */
-}
#endif /* MRB_INT64 */
static int
@@ -256,24 +239,17 @@ unpack_l(mrb_state *mrb, const unsigned char *src, int srclen, mrb_value ary, un
ul += (uint32_t)src[3];
}
if (flags & PACK_FLAG_SIGNED) {
- int32_t sl = ul;
-#ifndef MRB_INT64
- if (!FIXABLE(sl)) {
- i32tostr(msg, sizeof(msg), sl);
- mrb_raisef(mrb, E_RANGE_ERROR, "cannot unpack to Integer: %s", msg);
- }
-#endif
- n = sl;
+ n = (int32_t)ul;
} else {
#ifndef MRB_INT64
- if (!POSFIXABLE(ul)) {
+ if (UINT_OVERFLOW_P(ul)) {
u32tostr(msg, sizeof(msg), ul);
mrb_raisef(mrb, E_RANGE_ERROR, "cannot unpack to Integer: %s", msg);
}
#endif
n = ul;
}
- mrb_ary_push(mrb, ary, mrb_fixnum_value(n));
+ mrb_ary_push(mrb, ary, mrb_int_value(mrb, n));
return 4;
}
@@ -337,6 +313,7 @@ u64tostr(char *buf, size_t len, uint64_t n)
#endif /* MRB_NO_STDIO */
}
+#ifndef MRB_INT64
static void
i64tostr(char *buf, size_t len, int64_t n)
{
@@ -356,6 +333,7 @@ i64tostr(char *buf, size_t len, int64_t n)
snprintf(buf, len, "%" PRId64, n);
#endif /* MRB_NO_STDIO */
}
+#endif /* MRB_INT64 */
static int
unpack_q(mrb_state *mrb, const unsigned char *src, int srclen, mrb_value ary, unsigned int flags)
@@ -379,19 +357,21 @@ unpack_q(mrb_state *mrb, const unsigned char *src, int srclen, mrb_value ary, un
}
if (flags & PACK_FLAG_SIGNED) {
int64_t sll = ull;
- if (!FIXABLE(sll)) {
+#ifndef MRB_INT64
+ if (INT_OVERFLOW_P(sll)) {
i64tostr(msg, sizeof(msg), sll);
mrb_raisef(mrb, E_RANGE_ERROR, "cannot unpack to Integer: %s", msg);
}
+#endif
n = (mrb_int)sll;
} else {
- if (!POSFIXABLE(ull)) {
+ if (UINT_OVERFLOW_P(ull)) {
u64tostr(msg, sizeof(msg), ull);
mrb_raisef(mrb, E_RANGE_ERROR, "cannot unpack to Integer: %s", msg);
}
n = (mrb_int)ull;
}
- mrb_ary_push(mrb, ary, mrb_fixnum_value(n));
+ mrb_ary_push(mrb, ary, mrb_int_value(mrb, n));
return 8;
}