* thread.c (rb_thread_aref): add explanation for why Thread#[] and

Thread#[]= are fiber-local and not thread-local.
  reported by Julien A.  [ruby-core:41606] [ruby-trunk - Bug #5750]



git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@36269 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
akr 2012-07-02 13:15:29 +00:00
parent 83eceb2a79
commit c03d1c2736
2 changed files with 39 additions and 0 deletions

View File

@ -1,3 +1,9 @@
Mon Jul 2 22:13:04 2012 Tanaka Akira <akr@fsij.org>
* thread.c (rb_thread_aref): add explanation for why Thread#[] and
Thread#[]= are fiber-local and not thread-local.
reported by Julien A. [ruby-core:41606] [ruby-trunk - Bug #5750]
Mon Jul 2 21:25:55 2012 Tanaka Akira <akr@fsij.org>
* time.c (timew_out_of_timet_range): specialization for

View File

@ -2078,6 +2078,39 @@ rb_thread_local_aref(VALUE thread, ID id)
* #<Thread:0x00000002a54220 dead>: A
* #<Thread:0x00000002a541a8 dead>: B
* #<Thread:0x00000002a54130 dead>: C
*
* Thread#[] and Thread#[]= are not thread-local but fiber-local.
* This confusion was not exist until Ruby 1.8 because
* fiber is available since Ruby 1.9.
* Ruby 1.9 chooses that the methods behaves fiber-local to save
* following idiom for dynamic scope.
*
* def meth(newvalue)
* begin
* oldvalue = Thread.current[:name]
* Thread.current[:name] = newvalue
* yield
* ensure
* Thread.current[:name] = oldvalue
* end
* end
*
* The idiom may not work as dynamic scope if the methods are thread-local
* and a given block switches fiber.
*
* f = Fiber.new {
* meth(1) {
* Fiber.yield
* }
* }
* meth(2) {
* f.resume
* }
* f.resume
* p Thread.current[:name]
* #=> nil if fiber-local
* #=> 2 if thread-local (The value 2 is leaked to outside of meth method.)
*
*/
static VALUE