threaded, forked, rethreaded processes will deadlock
    Brian Fundakowski Feldman 
    green at freebsd.org
       
    Fri Jan  9 08:34:38 PST 2009
    
    
  
On Thu, Jan 08, 2009 at 11:09:16PM -0800, Julian Elischer wrote:
> Brian Fundakowski Feldman wrote:
>> On Thu, Jan 08, 2009 at 10:44:20PM -0500, Daniel Eischen wrote:
>>> On Thu, 8 Jan 2009, Brian Fundakowski Feldman wrote:
>>> 
>>>> It appears that the post-fork hooks for malloc(3) are somewhat broken such that
>>>> when a threaded program forks, and then its child attempts to go threaded, it
>>>> deadlocks because it already appears to have locks held.  I am not familiar
>>>> enough with the current libthr/libc/rtld-elf interaction that I've been able
>>>> to fix it myself, unfortunately.
>>> There's really nothing to fix - according to POSIX you are only
>>> allowed to call async-signal-safe functions in the child forked
>>> from a threaded process.  If you are trying to do anything other
>>> than that, it may or may not work on FreeBSD, but it is not
>>> guaranteed and is not portable.
>>> 
>>> The rationale is that what is the point of forking and creating
>>> more threads, when you can just as easily create more threads in
>>> the parent without forking?  The only reason to fork from a threaded
>>> process is to call one of the exec() functions.
>> 
>> Well, it worked until the last major set of changes to malloc.  For me, the point
>> was that I was able to have transparent background worker threads in any program
>> regardless of its architecture, using the standard pthread fork hooks.  Could you
>> point me to the POSIX section covering fork and threads?  If it's really not
>> supposed to work then that's fine, but there's an awful lot of code there dedicated
>> to support going threaded again after a fork.
>> 
> 
> Practically, you can't go threaded again after a fork
> (by which I mean creating new threads or use things
> like mutexes etc.) in any posix system I know of.
> 
> It would require that:
>  The forking thread stop until:
>   Every other thread has released every resource it owns
>   and reports itself to be in a "safe quiescent state",
>   or at least report every resource it owns, especially
>   locks,
>  and
>  After the fork:
>   The child, post fork, to take ownership of all
>   of them, and free them.
> 
> You might be able to do that in a simple
> threaded program, but consider then that the libraries may have
> threads running in them of which you are unaware, and that
> some of the resources may interract with resources owned by the
> forking thread.
> 
> Add to this that there may be a signal thrown into this mix as well
> 
> (signals are the bane of thread developement)
Well, I wouldn't mind showing all of you what I can of what I had been doing
with the background threads -- it works pretty well modulo this particular
malloc lock bug.  Due to it being inappropriate to share library resources
between a child and parent for an open socket connection, I always considered
the only "safe" behavior to be going single-threaded for the duration of the fork
processes in both the parent and child, and the pthread_atfork(3) hooks have been
sufficient to do so.
-- 
Brian Fundakowski Feldman                           \'[ FreeBSD ]''''''''''\
  <> green at FreeBSD.org                               \  The Power to Serve! \
 Opinions expressed are my own.                       \,,,,,,,,,,,,,,,,,,,,,,\
    
    
More information about the freebsd-hackers
mailing list