Project

General

Profile

Actions

Bug #20346

closed

FiberScheduler.unblock not called by Thread#join when Thread body contains Ractor.take

Bug #20346: FiberScheduler.unblock not called by Thread#join when Thread body contains Ractor.take

Added by forthoney (Seong-Heon Jung) over 1 year ago. Updated 2 months ago.

Status:
Closed
Assignee:
Target version:
-
[ruby-core:117216]

Description

When using a Ractor.take inside a different thread, Thread#join on the thread running Ractor.take fails to call FiberScheduler.unblock. The below code can replicate this behavior

require "async"
class RactorWrapper
  def initialize
    @ractor = Ractor.new do
      Ractor.recv # Ractor doesn't start until explicitly told to
      # Do some calculations
      fib = ->(x) { x < 2 ? 1 : fib.call(x - 1) + fib.call(x - 2) }
      fib.call(20)
    end
  end
  
  def take_async
    @ractor.send(nil)
    Thread.new { @ractor.take }.join.value
  end
end

Async do |task|
  10000.times do |i|
    task.async do
      RactorWrapper.new.take_async
      puts i
    end
  end
end

The above code deadlocks, and when we leave a debugging print statement inside of Async's scheduler's block and unblock method, we can confirm that we only call Scheduler.block, and never Scheduler.unblock

Updated by jpcamara (JP Camara) over 1 year ago Actions #1 [ruby-core:118335]

Looks as though the fiber scheduler is not supported by Ractors right now, due to fixes that need to be implemented in the Ractor API. Not clear if it is expected to be fixed for 3.4 or not.

Updated by jhawthorn (John Hawthorn) 6 months ago Actions #2

  • Assignee set to ractor

Updated by hsbt (Hiroshi SHIBATA) 5 months ago Actions #3

  • Status changed from Open to Assigned

Updated by jhawthorn (John Hawthorn) 2 months ago Actions #5 [ruby-core:123035]

  • Status changed from Assigned to Closed

This no longer deadlocks. Under the new Ractor API the take should be replaced with value.

require "async"

class RactorWrapper
  def initialize
    @ractor = Ractor.new do
      Ractor.recv # Ractor doesn't start until explicitly told to
      # Do some calculations
      fib = ->(x) { x < 2 ? 1 : fib.call(x - 1) + fib.call(x - 2) }
      fib.call(20)
    end
  end

  def take_async
    @ractor.send(nil)
    Thread.new { @ractor.value }.value
  end
end

Async do |task|
  10000.times do |i|
    task.async do
      RactorWrapper.new.take_async
      puts i
    end
  end
end

I recorded a profile showing that the concurrency is happening properly (warning: this has 10k threads of data) https://share.firefox.dev/4lCxnFK

Actions

Also available in: PDF Atom