Project

General

Profile

Actions

Bug #21031

closed

Incompatibility with prism and parse.y when eval'ing unnamed forwarding variables

Added by ksss (Yuki Kurihara) 5 days ago. Updated 3 days ago.

Status:
Closed
Assignee:
Target version:
-
ruby -v:
ruby 3.5.0dev (2025-01-13T03:12:28Z master 384e6945ac) +PRISM [arm64-darwin23]
[ruby-core:120628]

Description

Case 1

# t.rb
def foo(*)
  eval("p(*)")
end

foo(1)
$ ruby --parser=prism t.rb
t.rb:4:in 'Kernel#eval': (eval at t.rb:4):1: syntax error found (SyntaxError)
> 1 | p(*)
    |   ^ unexpected `*`; no anonymous rest parameter

	from t.rb:4:in 'Object#foo'
	from t.rb:7:in '<main>'
$ ruby --parser=parse.y t.rb
1

Case 2

# t.rb
def foo(**)
  eval("p(**)")
end

foo(a: 1)
$ ruby --parser=prism t.rb
t.rb:4:in 'Kernel#eval': (eval at t.rb:4):1: syntax error found (SyntaxError)
> 1 | p(**)
    |   ^~ unexpected `**`; no anonymous keyword rest parameter

	from t.rb:4:in 'Object#foo'
	from t.rb:7:in '<main>'
$ ruby --parser=parse.y t.rb
{a: 1}

Case 3

# t.rb
def foo(&)
  eval("p(&)")
end

foo{}
$ ruby --parser=prism t.rb
t.rb:4:in 'Kernel#eval': (eval at t.rb:4):1: syntax error found (SyntaxError)
> 1 | p(&)
    |   ^ unexpected `&`; no anonymous block parameter

	from t.rb:4:in 'Object#foo'
	from t.rb:7:in '<main>'
$ ruby --parser=parse.y t.rb

Case 4

# t.rb
def foo(...)
  eval("p(...)")
end

foo(1, 2, 3)
$ ruby --parser=prism t.rb
t.rb:4:in 'Kernel#eval': (eval at t.rb:4):1: syntax error found (SyntaxError)
> 1 | p(...)
    |   ^~~ unexpected ... when the parent method is not forwarding

	from t.rb:4:in 'Object#foo'
	from t.rb:7:in '<main>'
$ ruby --parser=parse.y t.rb
1
2
3

I expect to the parse.y behavior.

Updated by Eregon (Benoit Daloze) 5 days ago

@jeremyevans0 (Jeremy Evans) Isn't this parse.y behavior breaking the optimization to forward */**/& efficiently (and maybe ... too)?
It means static analysis can't reason about where the */**/&/... are used.

IMO all these should be SyntaxError.
I'm not sure the parse.y behavior is intentional.

Updated by Eregon (Benoit Daloze) 5 days ago · Edited

As a note, I checked and the * and ** examples (Case 1 & 2) work in 3.2 and 3.3.

Case 3 / & seems incorrect though, as it should print the block but doesn't.
EDIT: ah it's because p ignores the block and it's passed as a block and not a positional argument, nevermind.

Updated by jeremyevans0 (Jeremy Evans) 5 days ago

Eregon (Benoit Daloze) wrote in #note-1:

@jeremyevans0 (Jeremy Evans) Isn't this parse.y behavior breaking the optimization to forward */**/& efficiently (and maybe ... too)?
It means static analysis can't reason about where the */**/&/... are used.

For * and **, it shouldn't break the allocationless anonymous splat forwarding optimization. The optimization just makes the parameter not allocate an array/hash, regardless of the content of the method, since you can only splat the arguments, not access them directly.

For &, there is no optimization for anonymous block arguments. The block argument optimization to avoid Proc allocation was getblockparamproxy in Ruby 2.5, and applies to both named and anonymous blocks, and I believe works fine in eval (maybe @ko1 (Koichi Sasada) can confirm).

For ..., that never used allocationless anonymous splat forwarding in a shipped version of Ruby, only temporarily during the 3.4 development cycle. @tenderlovemaking (Aaron Patterson) applied a separate generic argument forwarding optimization (cdf33ed5f37f9649c482c3ba1d245f0d80ac01ce). I assume it works inside eval, but @tenderlovemaking (Aaron Patterson) would know better than I.

Updated by Eregon (Benoit Daloze) 5 days ago · Edited

  • Assignee set to prism

Thanks for clarifying, I thought the allocationless anonymous splat forwarding optimization relied on some specific usage of */** but it seems not, all good.

Then given this all works on 3.2 and 3.3 as well this seems a clear bug of the Prism compiler (and the lack of tests/specs for this).

Actions #5

Updated by kddnewton (Kevin Newton) 4 days ago

  • Status changed from Open to Closed

Applied in changeset git|cb419e3912f0514b8151469b0a4a4b83cbbcce78.


[PRISM] Handle forwarding inside eval

Fixes [Bug #21031]

Actions #6

Updated by kddnewton (Kevin Newton) 3 days ago

  • Backport changed from 3.1: UNKNOWN, 3.2: UNKNOWN, 3.3: UNKNOWN, 3.4: UNKNOWN to 3.1: UNKNOWN, 3.2: UNKNOWN, 3.3: UNKNOWN, 3.4: REQUIRED
Actions

Also available in: Atom PDF

Like0
Like0Like0Like0Like0Like0Like0