I'm implementing a simple windowing library as a Ruby C extension. Windows have a handle_events!
method that enters their native event loop.
The problem is that I want one event loop per window and the method blocks. I'd like the method to return immediately and let the loop run in a separate thread. What would be the best way to achieve this?
I tried using rb_thread_call_without_gvl
to call the event loop function, and then use rb_thread_call_with_gvl
in order to call the window's callbacks, which are Proc
s. Full source code can be found here.
It still works, but not as I intended: the method still blocks. Is this even possible with Ruby's threading model?
As far as I understand, using
rb_thread_call_with_gvl()
still needs to be done on the same thread. i.e.: it's about releasing and taking the global lock, not really about changing threads. For example, a long running gzip function can run without the lock so that other ruby threads can run in parallel.If you want your Procs called back on another thread, shouldn't you need to create a ruby thread for those Procs? Then on that thread, call out using
rb_thread_call_without_gvl()
to not hold the GVL (allowing other ruby threads to run), then when you have an event on the secondary window thread, callrb_thread_call_with_gvl()
to grab the lock and then you should be right to call the Proc on that same thread.That's the way I understand it... (not having done the C extension stuff very long.)