Skip to content

Deadlock in Concurrent::Array's on non-MRI #627

@eregon

Description

@eregon
Collaborator
  • concurrent-ruby version: 1.0.4
  • concurrent-ruby-ext installed: no
  • concurrent-ruby-edge used: no

Take the following code: https://gist.github.com/eregon/323890bfff539f8a33f66d8b2f02cc99
Of course, its purpose is to create a deadlock but it means any method taking a block on Array can cause a deadlock as long as:

  • 2+ threads use 2+ Concurrent::Array by calling a method taking a block
  • inside the blocks, the threads call any method taking on another Concurrent::Array

This seems not such a rare scenario, as it is frequent to call methods in a block that do not only involve the current Array.

The deadlock does not happen on MRI, as it uses a single global lock and concurrent-ruby just subclasses ::Array. It does happen on all other implementations though, like JRuby, Rubinius and TruffleRuby.

The documentation says:

A thread-safe subclass of Array.
This version locks against the object itself for every method call,
ensuring only one thread can be reading or writing at a time.
This includes iteration methods like #each.

So indeed this might imply the deadlock above, but it's not clear.
ensuring only one thread can be reading or writing at a time is also inaccurate on MRI for methods taking a block, as they release the GIL and might switch to another Thread in the middle of e.g. #each.

The same apply for Hash.

Is this behavior intended? Should this be fixed?
Should it behave the same on the different implementations?

Activity

eregon

eregon commented on Feb 2, 2017

@eregon
CollaboratorAuthor

Related to #594

added
bugA bug in the library or documentation.
on Feb 12, 2017
pitr-ch

pitr-ch commented on Feb 12, 2017

@pitr-ch
Member

I think this is a bug. 1) it should behave the same on all implementations. 2) it's a good practice not to execute user code when a lock is taken on a data structure, in this case to execute the each block while the block is taken

@thedarkone would you have time to work on this?

added this to the 1.1.0 milestone on Feb 12, 2017
modified the milestone: 1.1.1 on Apr 9, 2017
GustavoCaso

GustavoCaso commented on Nov 13, 2017

@GustavoCaso
Contributor

Hi @pitr-ch.
I'm a Ruby developer and I want to get more comfortable working with threads so I decided that the best way is by helping and understanding concurrent-ruby. I'm not very experienced with Threads is there any issue that is a good starting point?

pitr-ch

pitr-ch commented on Nov 18, 2017

@pitr-ch
Member

@GustavoCaso Thanks you very much for you interest. I'll welcome your help. Perhaps you could have a look at #670, it would give you opportunity to read the documentation for the given c-r parts and try it for yourself a with a replacement service you need to find (it can be anything not just finance) for fixing the doc.

modified the milestone: 1.1.1 on Feb 21, 2018
removed this from the 1.1.0 milestone on Jul 6, 2018
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Metadata

Metadata

Assignees

No one assigned

    Labels

    bugA bug in the library or documentation.looking-for-contributorWe are looking for a contributor to help with this issue.medium-priorityShould be done soon.

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

      Development

      No branches or pull requests

        Participants

        @eregon@pitr-ch@GustavoCaso

        Issue actions

          Deadlock in Concurrent::Array's on non-MRI · Issue #627 · ruby-concurrency/concurrent-ruby