summaryrefslogtreecommitdiffhomepage
path: root/src
diff options
context:
space:
mode:
authorksss <[email protected]>2016-09-08 16:18:48 +0900
committerksss <[email protected]>2016-09-08 16:35:17 +0900
commitd265c03da0a4a1f8fe3bd20d69c0f78588ec8aff (patch)
tree5030772d484a3623a8dd2847ea8fdb75ef8cc258 /src
parentcd5133c57065d9645c125c1ce7b5dc4ce6d39bdd (diff)
downloadmruby-d265c03da0a4a1f8fe3bd20d69c0f78588ec8aff.tar.gz
mruby-d265c03da0a4a1f8fe3bd20d69c0f78588ec8aff.zip
Fix SEGV when splat object
Splat operation should return an array. And raise an error if result of convert by to_a is not array or nil.
Diffstat (limited to 'src')
-rw-r--r--src/array.c23
1 files changed, 20 insertions, 3 deletions
diff --git a/src/array.c b/src/array.c
index 9b8a49584..df953832b 100644
--- a/src/array.c
+++ b/src/array.c
@@ -891,15 +891,32 @@ mrb_ary_rindex_m(mrb_state *mrb, mrb_value self)
MRB_API mrb_value
mrb_ary_splat(mrb_state *mrb, mrb_value v)
{
+ mrb_value a, recv_class;
+
if (mrb_array_p(v)) {
return v;
}
- if (mrb_respond_to(mrb, v, mrb_intern_lit(mrb, "to_a"))) {
- return mrb_funcall(mrb, v, "to_a", 0);
+
+ if (!mrb_respond_to(mrb, v, mrb_intern_lit(mrb, "to_a"))) {
+ return mrb_ary_new_from_values(mrb, 1, &v);
}
- else {
+
+ a = mrb_funcall(mrb, v, "to_a", 0);
+ if (mrb_array_p(a)) {
+ return a;
+ }
+ else if (mrb_nil_p(a)) {
return mrb_ary_new_from_values(mrb, 1, &v);
}
+ else {
+ recv_class = mrb_obj_value(mrb_obj_class(mrb, v));
+ mrb_raisef(mrb, E_TYPE_ERROR, "can't convert %S to Array (%S#to_a gives %S)",
+ recv_class,
+ recv_class,
+ mrb_obj_value(mrb_obj_class(mrb, a))
+ );
+ return mrb_undef_value();
+ }
}
static mrb_value