Commit be86d8b4 authored by Tomasz Dąbrowski's avatar Tomasz Dąbrowski

correctly handle *splat arguments in mrb_get_argc, also add mrb_vm_get_argc and mrb_get_argv

Fixes #3825
parent 80e03f3f
...@@ -858,12 +858,22 @@ mrb_get_mid(mrb_state *mrb) /* get method symbol */ ...@@ -858,12 +858,22 @@ mrb_get_mid(mrb_state *mrb) /* get method symbol */
return mrb->c->ci->mid; return mrb->c->ci->mid;
} }
/* returns -1 for *splat arguments */
static inline mrb_int 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; 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) /* `strlen` for character string literals (use with caution or `strlen` instead)
Adjacent string literals are concatenated in C/C++ in translation phase 6. Adjacent string literals are concatenated in C/C++ in translation phase 6.
If `lit` is not one, the compiler will report a syntax error: If `lit` is not one, the compiler will report a syntax error:
......
...@@ -532,6 +532,34 @@ to_sym(mrb_state *mrb, mrb_value ss) ...@@ -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. retrieve arguments from mrb_state.
...@@ -569,23 +597,14 @@ mrb_get_args(mrb_state *mrb, const char *format, ...) ...@@ -569,23 +597,14 @@ mrb_get_args(mrb_state *mrb, const char *format, ...)
char c; char c;
mrb_int i = 0; mrb_int i = 0;
va_list ap; va_list ap;
mrb_int argc = mrb->c->ci->argc; mrb_int argc = mrb_get_argc(mrb);
mrb_int arg_i = 0; mrb_int arg_i = 0;
mrb_value *array_argv; mrb_value *array_argv = mrb_get_argv(mrb);
mrb_bool opt = FALSE; mrb_bool opt = FALSE;
mrb_bool opt_skip = TRUE; mrb_bool opt_skip = TRUE;
mrb_bool given = TRUE; mrb_bool given = TRUE;
va_start(ap, format); 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 \ #define ARGV \
(array_argv ? array_argv : (mrb->c->stack + 1)) (array_argv ? array_argv : (mrb->c->stack + 1))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment