Project

General

Profile

Actions

Bug #14817

closed

TracePoint#parameters for bmethod's return event should return the same value as its Method#parameters

Added by ktsj (Kazuki Tsujimoto) over 6 years ago. Updated over 2 years ago.

Status:
Closed
Target version:
-
ruby -v:
ruby 2.6.0dev (2018-06-03 master 63562) [x86_64-linux]
[ruby-dev:50558]

Description

define_methodしたメソッド(bmethod)のcall/returnイベント中にTracePoint#parametersを呼び出すと
以下の結果となります。

$ cat t.rb
define_method(:bm) {|a|}

p method_parameters: method(:bm).parameters

trace = TracePoint.new(:call, :return){|tp|
  mid = tp.method_id
  if mid == :bm
    p mid: mid, event: tp.event, tp_parameters: tp.parameters
  end
}
trace.enable{
  bm(0)
}

$ ruby -v t.rb
ruby 2.6.0dev (2018-06-03 master 63562) [x86_64-linux]
{:method_parameters=>[[:req, :a]]}
{:mid=>:bm, :event=>:call, :tp_parameters=>[[:req, :a]]}
{:mid=>:bm, :event=>:return, :tp_parameters=>[]}         #=> expected: {:mid=>:bm, :event=>:return, :tp_parameters=>[[:req, :a]]}

現状、callイベントに限ってその戻り値がbmethodのMethod#parametersの呼び出し結果と一致しますが
returnイベント時も同様となるべきだと思います。


Related issues 1 (0 open1 closed)

Related to Ruby master - Bug #13392: TracePoint return event location is incorrect for methods defined with define_methodClosedActions

Updated by mame (Yusuke Endoh) over 6 years ago

TracePoint#parameters の問題ではなく、define_method + TracePoint 全般の問題のようです。

define_method(:bm) {|a|}

trace = TracePoint.new(:call, :return){|tp|
  p [tp.event, tp.lineno] if tp.method_id == :bm
}
trace.enable{
  bm(0)
}
$ ./miniruby test.rb
[:call, 1]
[:return, 7] #=> [:return, 1] になるべき?

Updated by mame (Yusuke Endoh) over 6 years ago

  • Assignee changed from mame (Yusuke Endoh) to ko1 (Koichi Sasada)

invoke_bmethod で vm_exec が終わったあとに EXEC_EVENT_HOOK(ec, RUBY_EVENT_RETURN, ...) するため、このような挙動になっているようです。
これは仕様でしょうか。とりあえずささださんに振ります。

Updated by ko1 (Koichi Sasada) about 6 years ago

見逃してました。

これ、実装見てたら、どーしょーもない気がするんで、制限のある仕様、ってことで駄目でしょうか。駄目かなあ。実装アイディア募集。

Actions #4

Updated by ko1 (Koichi Sasada) about 6 years ago

  • Status changed from Assigned to Feedback

Updated by ko1 (Koichi Sasada) over 5 years ago

2.6では known issue にします。
2.7 で、define_method したら ISeq 作るようにするとどうかな、と思っているので、それで一気に解決します。

Actions #6

Updated by ko1 (Koichi Sasada) over 5 years ago

  • Target version changed from 2.6 to 2.7
Actions #7

Updated by naruse (Yui NARUSE) over 5 years ago

  • Target version deleted (2.7)

Updated by ko1 (Koichi Sasada) about 5 years ago

  • Status changed from Feedback to Assigned

2.7 で、define_method したら ISeq 作るようにするとどうかな、と思っているので、それで一気に解決します。

これでうまくいかない、ということがわかったので、ちょっと宙ぶらりんです。

考えていた解決案:

  • (1) proc を受け取る
  • (2) proc のパラメータとまったく同じ method iseq を作る
  • (3) 受け取ったパラメータを proc に渡す

こんな感じです。

foo_body = proc{|a, b| xxx}
def foo(a, b) # foo_body.parameters を見て、パラメータリストを作る
  foo_body[a, b]
end

これでいけるじゃん、天才だなと思ってたんですが、optional 引数 o = expr が入ると、うまくいかないことがわかりました。optional 引数の分を、rest で受け取っちゃう、ってのも手だけど、それだと実際に rest が居たときに面倒そうかなぁ。ああ、foo の method parameter が rest になっちゃうから良くないですね。

この問題について、解決案が全然思い浮かびません。

Actions #9

Updated by jeremyevans0 (Jeremy Evans) about 3 years ago

  • Related to Bug #13392: TracePoint return event location is incorrect for methods defined with define_method added

Updated by jeremyevans0 (Jeremy Evans) about 3 years ago

The cause of this issue is the same as #13392. I've verified the pull request I submitted for that issue (https://github.com/ruby/ruby/pull/4588) will also fix this issue.

Updated by ko1 (Koichi Sasada) over 2 years ago

  • Status changed from Assigned to Closed
Actions

Also available in: Atom PDF

Like0
Like0Like0Like0Like0Like0Like0Like0Like0Like0Like0Like0