aboutsummaryrefslogtreecommitdiffstats
path: root/include/ruby
diff options
context:
space:
mode:
authornobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2017-08-20 06:08:25 +0000
committernobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2017-08-20 06:08:25 +0000
commit1d02f94c2ecf47e992b49357c13e04b53d39c5e3 (patch)
treebf5e07f2f8b39dca7872ed712df6a92686d339c6 /include/ruby
parent6a61b21e5af9e5255c500d5033e7e525062bcd4d (diff)
downloadruby-1d02f94c2ecf47e992b49357c13e04b53d39c5e3.tar.gz
non-keywords hash
* class.c (rb_scan_args), include/ruby/ruby.h (rb_scan_args_set): return non-keywords elements only in the last hash when keyword arguments are extracted from it, as well as methods defined in ruby level. [ruby-core:82427] [Bug #13830] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@59626 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'include/ruby')
-rw-r--r--include/ruby/ruby.h19
1 files changed, 12 insertions, 7 deletions
diff --git a/include/ruby/ruby.h b/include/ruby/ruby.h
index 566e5f6b32..43a5f6fdeb 100644
--- a/include/ruby/ruby.h
+++ b/include/ruby/ruby.h
@@ -2351,8 +2351,8 @@ rb_scan_args_set(int argc, const VALUE *argv,
int f_var, int f_hash, int f_block,
VALUE *vars[])
{
- int i, argi = 0, vari = 0;
- VALUE *var, hash = Qnil;
+ int i, argi = 0, vari = 0, last_idx = -1;
+ VALUE *var, hash = Qnil, last_hash = 0;
const int n_mand = n_lead + n_trail;
/* capture an option hash - phase 1: pop */
@@ -2370,7 +2370,8 @@ rb_scan_args_set(int argc, const VALUE *argv,
hash = rb_check_hash_type(last);
if (!RB_NIL_P(hash)) {
VALUE opts = rb_extract_keywords(&hash);
- if (!hash) argc--;
+ if (!(last_hash = hash)) argc--;
+ else last_idx = argc - 1;
hash = opts ? opts : Qnil;
}
}
@@ -2381,14 +2382,14 @@ rb_scan_args_set(int argc, const VALUE *argv,
/* capture leading mandatory arguments */
for (i = n_lead; i-- > 0; ) {
var = vars[vari++];
- if (var) *var = argv[argi];
+ if (var) *var = (argi == last_idx) ? last_hash : argv[argi];
argi++;
}
/* capture optional arguments */
for (i = n_opt; i-- > 0; ) {
var = vars[vari++];
if (argi < argc - n_trail) {
- if (var) *var = argv[argi];
+ if (var) *var = (argi == last_idx) ? last_hash : argv[argi];
argi++;
}
else {
@@ -2401,7 +2402,11 @@ rb_scan_args_set(int argc, const VALUE *argv,
var = vars[vari++];
if (0 < n_var) {
- if (var) *var = rb_ary_new4(n_var, &argv[argi]);
+ if (var) {
+ int f_last = (last_idx + 1 == argc - n_trail);
+ *var = rb_ary_new4(n_var-f_last, &argv[argi]);
+ if (f_last) rb_ary_push(*var, last_hash);
+ }
argi += n_var;
}
else {
@@ -2411,7 +2416,7 @@ rb_scan_args_set(int argc, const VALUE *argv,
/* capture trailing mandatory arguments */
for (i = n_trail; i-- > 0; ) {
var = vars[vari++];
- if (var) *var = argv[argi];
+ if (var) *var = (argi == last_idx) ? last_hash : argv[argi];
argi++;
}
/* capture an option hash - phase 2: assignment */