Add three more C-API functions for handling keywords
This adds rb_funcall_passing_block_kw, rb_funcallv_public_kw, and rb_yield_splat_kw. This functions are necessary to easily handle cases where rb_funcall_passing_block, rb_funcallv_public, and rb_yield_splat are currently used and a keyword argument separation warning is raised.
This commit is contained in:
parent
fba8627dc1
commit
649a64ae29
Notes:
git
2019-09-30 10:31:30 +09:00
@ -1,7 +1,5 @@
|
|||||||
#include "ruby.h"
|
#include "ruby.h"
|
||||||
|
|
||||||
VALUE rb_funcall_passing_block(VALUE, ID, int, const VALUE*);
|
|
||||||
|
|
||||||
static VALUE
|
static VALUE
|
||||||
with_funcall2(int argc, VALUE *argv, VALUE self)
|
with_funcall2(int argc, VALUE *argv, VALUE self)
|
||||||
{
|
{
|
||||||
@ -14,6 +12,24 @@ with_funcall_passing_block(int argc, VALUE *argv, VALUE self)
|
|||||||
return rb_funcall_passing_block(self, rb_intern("target"), argc, argv);
|
return rb_funcall_passing_block(self, rb_intern("target"), argc, argv);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static VALUE
|
||||||
|
with_funcall_passing_block_kw(int argc, VALUE *argv, VALUE self)
|
||||||
|
{
|
||||||
|
return rb_funcall_passing_block_kw(self, rb_intern("target"), argc-1, argv+1, FIX2INT(argv[0]));
|
||||||
|
}
|
||||||
|
|
||||||
|
static VALUE
|
||||||
|
with_funcallv_public_kw(int argc, VALUE *argv, VALUE self)
|
||||||
|
{
|
||||||
|
return rb_funcallv_public_kw(argv[0], SYM2ID(argv[1]), argc-3, argv+3, FIX2INT(argv[2]));
|
||||||
|
}
|
||||||
|
|
||||||
|
static VALUE
|
||||||
|
with_yield_splat_kw(int argc, VALUE *argv, VALUE self)
|
||||||
|
{
|
||||||
|
return rb_yield_splat_kw(argv[1], FIX2INT(argv[0]));
|
||||||
|
}
|
||||||
|
|
||||||
static VALUE
|
static VALUE
|
||||||
extra_args_name(VALUE self)
|
extra_args_name(VALUE self)
|
||||||
{
|
{
|
||||||
@ -34,10 +50,22 @@ Init_funcall(void)
|
|||||||
"with_funcall2",
|
"with_funcall2",
|
||||||
with_funcall2,
|
with_funcall2,
|
||||||
-1);
|
-1);
|
||||||
|
rb_define_singleton_method(cRelay,
|
||||||
|
"with_funcall_passing_block_kw",
|
||||||
|
with_funcall_passing_block_kw,
|
||||||
|
-1);
|
||||||
rb_define_singleton_method(cRelay,
|
rb_define_singleton_method(cRelay,
|
||||||
"with_funcall_passing_block",
|
"with_funcall_passing_block",
|
||||||
with_funcall_passing_block,
|
with_funcall_passing_block,
|
||||||
-1);
|
-1);
|
||||||
|
rb_define_singleton_method(cRelay,
|
||||||
|
"with_funcallv_public_kw",
|
||||||
|
with_funcallv_public_kw,
|
||||||
|
-1);
|
||||||
|
rb_define_singleton_method(cRelay,
|
||||||
|
"with_yield_splat_kw",
|
||||||
|
with_yield_splat_kw,
|
||||||
|
-1);
|
||||||
rb_define_singleton_method(cTestFuncall, "extra_args_name",
|
rb_define_singleton_method(cTestFuncall, "extra_args_name",
|
||||||
extra_args_name,
|
extra_args_name,
|
||||||
0);
|
0);
|
||||||
|
@ -1893,9 +1893,11 @@ VALUE rb_funcall(VALUE, ID, int, ...);
|
|||||||
VALUE rb_funcallv(VALUE, ID, int, const VALUE*);
|
VALUE rb_funcallv(VALUE, ID, int, const VALUE*);
|
||||||
VALUE rb_funcallv_kw(VALUE, ID, int, const VALUE*, int);
|
VALUE rb_funcallv_kw(VALUE, ID, int, const VALUE*, int);
|
||||||
VALUE rb_funcallv_public(VALUE, ID, int, const VALUE*);
|
VALUE rb_funcallv_public(VALUE, ID, int, const VALUE*);
|
||||||
|
VALUE rb_funcallv_public_kw(VALUE, ID, int, const VALUE*, int);
|
||||||
#define rb_funcall2 rb_funcallv
|
#define rb_funcall2 rb_funcallv
|
||||||
#define rb_funcall3 rb_funcallv_public
|
#define rb_funcall3 rb_funcallv_public
|
||||||
VALUE rb_funcall_passing_block(VALUE, ID, int, const VALUE*);
|
VALUE rb_funcall_passing_block(VALUE, ID, int, const VALUE*);
|
||||||
|
VALUE rb_funcall_passing_block_kw(VALUE, ID, int, const VALUE*, int);
|
||||||
VALUE rb_funcall_with_block(VALUE, ID, int, const VALUE*, VALUE);
|
VALUE rb_funcall_with_block(VALUE, ID, int, const VALUE*, VALUE);
|
||||||
VALUE rb_funcall_with_block_kw(VALUE, ID, int, const VALUE*, VALUE, int);
|
VALUE rb_funcall_with_block_kw(VALUE, ID, int, const VALUE*, VALUE, int);
|
||||||
int rb_scan_args(int, const VALUE*, const char*, ...);
|
int rb_scan_args(int, const VALUE*, const char*, ...);
|
||||||
@ -1972,6 +1974,7 @@ VALUE rb_yield_values(int n, ...);
|
|||||||
VALUE rb_yield_values2(int n, const VALUE *argv);
|
VALUE rb_yield_values2(int n, const VALUE *argv);
|
||||||
VALUE rb_yield_values_kw(int n, const VALUE *argv, int kw_splat);
|
VALUE rb_yield_values_kw(int n, const VALUE *argv, int kw_splat);
|
||||||
VALUE rb_yield_splat(VALUE);
|
VALUE rb_yield_splat(VALUE);
|
||||||
|
VALUE rb_yield_splat_kw(VALUE, int);
|
||||||
VALUE rb_yield_block(RB_BLOCK_CALL_FUNC_ARGLIST(yielded_arg, callback_arg)); /* rb_block_call_func */
|
VALUE rb_yield_block(RB_BLOCK_CALL_FUNC_ARGLIST(yielded_arg, callback_arg)); /* rb_block_call_func */
|
||||||
#define RB_NO_KEYWORDS 0
|
#define RB_NO_KEYWORDS 0
|
||||||
#define RB_PASS_KEYWORDS 1
|
#define RB_PASS_KEYWORDS 1
|
||||||
|
@ -3,8 +3,8 @@ require 'test/unit'
|
|||||||
|
|
||||||
class TestFuncall < Test::Unit::TestCase
|
class TestFuncall < Test::Unit::TestCase
|
||||||
module Relay
|
module Relay
|
||||||
def self.target(*args, &block)
|
def self.target(*args, **kw, &block)
|
||||||
yield(*args) if block
|
yield(*args, **kw) if block
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
require '-test-/funcall'
|
require '-test-/funcall'
|
||||||
@ -20,4 +20,56 @@ class TestFuncall < Test::Unit::TestCase
|
|||||||
Relay.with_funcall_passing_block("feature#4504") {|arg| ok = arg || true}
|
Relay.with_funcall_passing_block("feature#4504") {|arg| ok = arg || true}
|
||||||
assert_equal("feature#4504", ok)
|
assert_equal("feature#4504", ok)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_with_funcall_passing_block_kw
|
||||||
|
block = ->(*a, **kw) { [a, kw] }
|
||||||
|
assert_equal([[1], {}], Relay.with_funcall_passing_block_kw(0, 1, &block))
|
||||||
|
assert_equal([[], {a: 1}], Relay.with_funcall_passing_block_kw(1, a: 1, &block))
|
||||||
|
assert_equal([[1], {a: 1}], Relay.with_funcall_passing_block_kw(1, 1, a: 1, &block))
|
||||||
|
assert_equal([[{}], {}], Relay.with_funcall_passing_block_kw(2, {}, **{}, &block))
|
||||||
|
assert_equal([[], {a: 1}], Relay.with_funcall_passing_block_kw(3, a: 1, &block))
|
||||||
|
assert_equal([[{a: 1}], {}], Relay.with_funcall_passing_block_kw(3, {a: 1}, **{}, &block))
|
||||||
|
assert_warn(/warning: The keyword argument is passed as the last hash parameter.*for method/m) do
|
||||||
|
assert_equal({}, Relay.with_funcall_passing_block_kw(3, **{}, &->(a){a}))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_with_funcallv_public_kw
|
||||||
|
o = Object.new
|
||||||
|
def o.foo(*args, **kw)
|
||||||
|
[args, kw]
|
||||||
|
end
|
||||||
|
def o.bar(*args, **kw)
|
||||||
|
[args, kw]
|
||||||
|
end
|
||||||
|
o.singleton_class.send(:private, :bar)
|
||||||
|
def o.baz(arg)
|
||||||
|
arg
|
||||||
|
end
|
||||||
|
assert_equal([[1], {}], Relay.with_funcallv_public_kw(o, :foo, 0, 1))
|
||||||
|
assert_equal([[], {a: 1}], Relay.with_funcallv_public_kw(o, :foo, 1, a: 1))
|
||||||
|
assert_equal([[1], {a: 1}], Relay.with_funcallv_public_kw(o, :foo, 1, 1, a: 1))
|
||||||
|
assert_equal([[{}], {}], Relay.with_funcallv_public_kw(o, :foo, 2, {}, **{}))
|
||||||
|
assert_equal([[], {a: 1}], Relay.with_funcallv_public_kw(o, :foo, 3, a: 1))
|
||||||
|
assert_equal([[{a: 1}], {}], Relay.with_funcallv_public_kw(o, :foo, 3, {a: 1}, **{}))
|
||||||
|
assert_raise(NoMethodError) { Relay.with_funcallv_public_kw(o, :bar, 3, {a: 1}, **{}) }
|
||||||
|
assert_warn(/warning: The keyword argument is passed as the last hash parameter.*for `baz'/m) do
|
||||||
|
assert_equal({}, Relay.with_funcallv_public_kw(o, :baz, 3, **{}))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_with_yield_splat_kw
|
||||||
|
block = ->(*a, **kw) { [a, kw] }
|
||||||
|
assert_equal([[1], {}], Relay.with_yield_splat_kw(0, [1], &block))
|
||||||
|
assert_equal([[], {a: 1}], Relay.with_yield_splat_kw(1, [{a: 1}], &block))
|
||||||
|
assert_equal([[1], {a: 1}], Relay.with_yield_splat_kw(1, [1, {a: 1}], &block))
|
||||||
|
assert_equal([[{}], {}], Relay.with_yield_splat_kw(2, [{}], **{}, &block))
|
||||||
|
assert_warn(/warning: The last argument is used as the keyword parameter.*for method/m) do
|
||||||
|
assert_equal([[], {a: 1}], Relay.with_yield_splat_kw(3, [{a: 1}], &block))
|
||||||
|
end
|
||||||
|
assert_equal([[{a: 1}], {}], Relay.with_yield_splat_kw(3, [{a: 1}], **{}, &block))
|
||||||
|
assert_warn(/warning: The keyword argument is passed as the last hash parameter/) do
|
||||||
|
assert_equal({}, Relay.with_yield_splat_kw(3, [], **{}, &->(a){a}))
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
33
vm_eval.c
33
vm_eval.c
@ -989,6 +989,15 @@ rb_funcallv_public(VALUE recv, ID mid, int argc, const VALUE *argv)
|
|||||||
return rb_call(recv, mid, argc, argv, CALL_PUBLIC);
|
return rb_call(recv, mid, argc, argv, CALL_PUBLIC);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VALUE
|
||||||
|
rb_funcallv_public_kw(VALUE recv, ID mid, int argc, const VALUE *argv, int kw_splat)
|
||||||
|
{
|
||||||
|
VALUE v = rb_adjust_argv_kw_splat(&argc, &argv, &kw_splat);
|
||||||
|
VALUE ret = rb_call(recv, mid, argc, argv, kw_splat ? CALL_PUBLIC_KW : CALL_PUBLIC);
|
||||||
|
rb_free_tmp_buffer(&v);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Calls a method
|
* Calls a method
|
||||||
* \private
|
* \private
|
||||||
@ -1032,6 +1041,17 @@ rb_funcall_passing_block(VALUE recv, ID mid, int argc, const VALUE *argv)
|
|||||||
return rb_call(recv, mid, argc, argv, CALL_PUBLIC);
|
return rb_call(recv, mid, argc, argv, CALL_PUBLIC);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VALUE
|
||||||
|
rb_funcall_passing_block_kw(VALUE recv, ID mid, int argc, const VALUE *argv, int kw_splat)
|
||||||
|
{
|
||||||
|
VALUE v = rb_adjust_argv_kw_splat(&argc, &argv, &kw_splat);
|
||||||
|
VALUE ret;
|
||||||
|
PASS_PASSED_BLOCK_HANDLER();
|
||||||
|
ret = rb_call(recv, mid, argc, argv, kw_splat ? CALL_PUBLIC_KW : CALL_PUBLIC);
|
||||||
|
rb_free_tmp_buffer(&v);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
VALUE
|
VALUE
|
||||||
rb_funcall_with_block(VALUE recv, ID mid, int argc, const VALUE *argv, VALUE passed_procval)
|
rb_funcall_with_block(VALUE recv, ID mid, int argc, const VALUE *argv, VALUE passed_procval)
|
||||||
{
|
{
|
||||||
@ -1278,6 +1298,19 @@ rb_yield_splat(VALUE values)
|
|||||||
return v;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VALUE
|
||||||
|
rb_yield_splat_kw(VALUE values, int kw_splat)
|
||||||
|
{
|
||||||
|
VALUE tmp = rb_check_array_type(values);
|
||||||
|
VALUE v;
|
||||||
|
if (NIL_P(tmp)) {
|
||||||
|
rb_raise(rb_eArgError, "not an array");
|
||||||
|
}
|
||||||
|
v = rb_yield_0_kw(RARRAY_LENINT(tmp), RARRAY_CONST_PTR(tmp), kw_splat);
|
||||||
|
RB_GC_GUARD(tmp);
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
|
||||||
VALUE
|
VALUE
|
||||||
rb_yield_force_blockarg(VALUE values)
|
rb_yield_force_blockarg(VALUE values)
|
||||||
{
|
{
|
||||||
|
Loading…
x
Reference in New Issue
Block a user