summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorTomasz Dąbrowski <[email protected]>2017-10-10 18:40:38 +0200
committerTomasz Dąbrowski <[email protected]>2017-10-10 18:40:38 +0200
commitbe86d8b45fa5d83f1d1caba9c4c0893c6281f720 (patch)
tree6a93866751d0733471666b198a9ac0b82a4aa721
parent80e03f3ffb40aa1cf6d40c1305fcd913bf82ab1d (diff)
downloadmruby-be86d8b45fa5d83f1d1caba9c4c0893c6281f720.tar.gz
mruby-be86d8b45fa5d83f1d1caba9c4c0893c6281f720.zip
correctly handle *splat arguments in mrb_get_argc, also add mrb_vm_get_argc and mrb_get_argv
Fixes #3825
-rw-r--r--include/mruby.h12
-rw-r--r--src/class.c41
2 files changed, 41 insertions, 12 deletions
diff --git a/include/mruby.h b/include/mruby.h
index 316707909..54ba0ead4 100644
--- a/include/mruby.h
+++ b/include/mruby.h
@@ -858,12 +858,22 @@ mrb_get_mid(mrb_state *mrb) /* get method symbol */
return mrb->c->ci->mid;
}
+/* returns -1 for *splat arguments */
static inline mrb_int
-mrb_get_argc(mrb_state *mrb) /* get argc */
+mrb_vm_get_argc(mrb_state *mrb) /* get argc */
{
return mrb->c->ci->argc;
}
+/**
+ * Retrieve number of arguments from mrb_state.
+ *
+ * Correctly handles *splat arguments.
+ */
+MRB_API mrb_int mrb_get_argc(mrb_state *mrb);
+
+MRB_API mrb_value* mrb_get_argv(mrb_state *mrb);
+
/* `strlen` for character string literals (use with caution or `strlen` instead)
Adjacent string literals are concatenated in C/C++ in translation phase 6.
If `lit` is not one, the compiler will report a syntax error:
diff --git a/src/class.c b/src/class.c
index f220106cd..506f540af 100644
--- a/src/class.c
+++ b/src/class.c
@@ -532,6 +532,34 @@ to_sym(mrb_state *mrb, mrb_value ss)
}
}
+MRB_API mrb_int
+mrb_get_argc(mrb_state *mrb)
+{
+ mrb_int argc = mrb_vm_get_argc(mrb);
+ if (argc < 0) {
+ struct RArray *a = mrb_ary_ptr(mrb->c->stack[1]);
+
+ argc = ARY_LEN(a);
+ }
+ return argc;
+}
+
+MRB_API mrb_value*
+mrb_get_argv(mrb_state *mrb)
+{
+ mrb_int argc = mrb_vm_get_argc(mrb);
+ mrb_value *array_argv;
+ if (argc < 0) {
+ struct RArray *a = mrb_ary_ptr(mrb->c->stack[1]);
+
+ array_argv = ARY_PTR(a);
+ }
+ else {
+ array_argv = NULL;
+ }
+ return array_argv;
+}
+
/*
retrieve arguments from mrb_state.
@@ -569,23 +597,14 @@ mrb_get_args(mrb_state *mrb, const char *format, ...)
char c;
mrb_int i = 0;
va_list ap;
- mrb_int argc = mrb->c->ci->argc;
+ mrb_int argc = mrb_get_argc(mrb);
mrb_int arg_i = 0;
- mrb_value *array_argv;
+ mrb_value *array_argv = mrb_get_argv(mrb);
mrb_bool opt = FALSE;
mrb_bool opt_skip = TRUE;
mrb_bool given = TRUE;
va_start(ap, format);
- if (argc < 0) {
- struct RArray *a = mrb_ary_ptr(mrb->c->stack[1]);
-
- argc = ARY_LEN(a);
- array_argv = ARY_PTR(a);
- }
- else {
- array_argv = NULL;
- }
#define ARGV \
(array_argv ? array_argv : (mrb->c->stack + 1))