Bug #21031
closedIncompatibility with prism and parse.y when eval'ing unnamed forwarding variables
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).
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]
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