olga / rpms / glibc

Forked from rpms/glibc 5 years ago
Clone

Blame SOURCES/glibc-rh906468-1.patch

8ae002
Backport of these upstream commits:
8ae002
8ae002
commit 29d794863cd6e03115d3670707cc873a9965ba92
8ae002
Author: Florian Weimer <fweimer@redhat.com>
8ae002
Date:   Thu Apr 14 09:17:02 2016 +0200
8ae002
8ae002
    malloc: Run fork handler as late as possible [BZ #19431]
8ae002
    
8ae002
    Previously, a thread M invoking fork would acquire locks in this order:
8ae002
    
8ae002
      (M1) malloc arena locks (in the registered fork handler)
8ae002
      (M2) libio list lock
8ae002
    
8ae002
    A thread F invoking flush (NULL) would acquire locks in this order:
8ae002
    
8ae002
      (F1) libio list lock
8ae002
      (F2) individual _IO_FILE locks
8ae002
    
8ae002
    A thread G running getdelim would use this order:
8ae002
    
8ae002
      (G1) _IO_FILE lock
8ae002
      (G2) malloc arena lock
8ae002
    
8ae002
    After executing (M1), (F1), (G1), none of the threads can make progress.
8ae002
    
8ae002
    This commit changes the fork lock order to:
8ae002
    
8ae002
      (M'1) libio list lock
8ae002
      (M'2) malloc arena locks
8ae002
    
8ae002
    It explicitly encodes the lock order in the implementations of fork,
8ae002
    and does not rely on the registration order, thus avoiding the deadlock.
8ae002
8ae002
commit 186fe877f3df0b84d57dfbf0386f6332c6aa69bc
8ae002
Author: Florian Weimer <fweimer@redhat.com>
8ae002
Date:   Thu Apr 14 12:53:03 2016 +0200
8ae002
8ae002
    malloc: Add missing internal_function attributes on function definitions
8ae002
    
8ae002
    Fixes build on i386 after commit 29d794863cd6e03115d3670707cc873a9965ba92.
8ae002
8ae002
Index: b/malloc/Makefile
8ae002
===================================================================
8ae002
--- a/malloc/Makefile
8ae002
+++ b/malloc/Makefile
8ae002
@@ -28,7 +28,7 @@ tests := mallocbug tst-malloc tst-valloc
8ae002
 	 tst-mallocstate tst-mcheck tst-mallocfork tst-trim1 \
8ae002
 	 tst-malloc-usable \
8ae002
 	 tst-malloc-backtrace tst-malloc-thread-exit \
8ae002
-	 tst-malloc-thread-fail
8ae002
+	 tst-malloc-thread-fail tst-malloc-fork-deadlock
8ae002
 test-srcs = tst-mtrace
8ae002
 
8ae002
 routines = malloc morecore mcheck mtrace obstack
8ae002
@@ -49,6 +49,7 @@ $(objpfx)tst-malloc-thread-fail: $(commo
8ae002
 			       $(common-objpfx)nptl/libpthread_nonshared.a
8ae002
 $(objpfx)tst-malloc-thread-exit: $(common-objpfx)nptl/libpthread.so \
8ae002
 			       $(common-objpfx)nptl/libpthread_nonshared.a
8ae002
+$(objpfx)tst-malloc-fork-deadlock: $(shared-thread-library)
8ae002
 
8ae002
 # These should be removed by `make clean'.
8ae002
 extra-objs = mcheck-init.o libmcheck.a
8ae002
Index: b/malloc/arena.c
8ae002
===================================================================
8ae002
--- a/malloc/arena.c
8ae002
+++ b/malloc/arena.c
8ae002
@@ -162,10 +162,6 @@ static void           (*save_free_hook)
8ae002
 					 const __malloc_ptr_t);
8ae002
 static void*        save_arena;
8ae002
 
8ae002
-#ifdef ATFORK_MEM
8ae002
-ATFORK_MEM;
8ae002
-#endif
8ae002
-
8ae002
 /* Magic value for the thread-specific arena pointer when
8ae002
    malloc_atfork() is in use.  */
8ae002
 
8ae002
@@ -228,14 +224,15 @@ free_atfork(void* mem, const void *calle
8ae002
 /* Counter for number of times the list is locked by the same thread.  */
8ae002
 static unsigned int atfork_recursive_cntr;
8ae002
 
8ae002
-/* The following two functions are registered via thread_atfork() to
8ae002
-   make sure that the mutexes remain in a consistent state in the
8ae002
-   fork()ed version of a thread.  Also adapt the malloc and free hooks
8ae002
-   temporarily, because the `atfork' handler mechanism may use
8ae002
-   malloc/free internally (e.g. in LinuxThreads). */
8ae002
+/* The following three functions are called around fork from a
8ae002
+   multi-threaded process.  We do not use the general fork handler
8ae002
+   mechanism to make sure that our handlers are the last ones being
8ae002
+   called, so that other fork handlers can use the malloc
8ae002
+   subsystem.  */
8ae002
 
8ae002
-static void
8ae002
-ptmalloc_lock_all (void)
8ae002
+void
8ae002
+internal_function
8ae002
+__malloc_fork_lock_parent (void)
8ae002
 {
8ae002
   mstate ar_ptr;
8ae002
 
8ae002
@@ -243,7 +240,7 @@ ptmalloc_lock_all (void)
8ae002
     return;
8ae002
 
8ae002
   /* We do not acquire free_list_lock here because we completely
8ae002
-     reconstruct free_list in ptmalloc_unlock_all2.  */
8ae002
+     reconstruct free_list in __malloc_fork_unlock_child.  */
8ae002
 
8ae002
   if (mutex_trylock(&list_lock))
8ae002
     {
8ae002
@@ -268,7 +265,7 @@ ptmalloc_lock_all (void)
8ae002
   __free_hook = free_atfork;
8ae002
   /* Only the current thread may perform malloc/free calls now.
8ae002
      save_arena will be reattached to the current thread, in
8ae002
-     ptmalloc_lock_all, so save_arena->attached_threads is not
8ae002
+     __malloc_fork_lock_parent, so save_arena->attached_threads is not
8ae002
      updated.  */
8ae002
   tsd_getspecific(arena_key, save_arena);
8ae002
   tsd_setspecific(arena_key, ATFORK_ARENA_PTR);
8ae002
@@ -276,8 +273,9 @@ ptmalloc_lock_all (void)
8ae002
   ++atfork_recursive_cntr;
8ae002
 }
8ae002
 
8ae002
-static void
8ae002
-ptmalloc_unlock_all (void)
8ae002
+void
8ae002
+internal_function
8ae002
+__malloc_fork_unlock_parent (void)
8ae002
 {
8ae002
   mstate ar_ptr;
8ae002
 
8ae002
@@ -286,8 +284,8 @@ ptmalloc_unlock_all (void)
8ae002
   if (--atfork_recursive_cntr != 0)
8ae002
     return;
8ae002
   /* Replace ATFORK_ARENA_PTR with save_arena.
8ae002
-     save_arena->attached_threads was not changed in ptmalloc_lock_all
8ae002
-     and is still correct.  */
8ae002
+     save_arena->attached_threads was not changed in
8ae002
+     __malloc_fork_lock_parent and is still correct.  */
8ae002
   tsd_setspecific(arena_key, save_arena);
8ae002
   __malloc_hook = save_malloc_hook;
8ae002
   __free_hook = save_free_hook;
8ae002
@@ -299,15 +297,9 @@ ptmalloc_unlock_all (void)
8ae002
   (void)mutex_unlock(&list_lock);
8ae002
 }
8ae002
 
8ae002
-# ifdef __linux__
8ae002
-
8ae002
-/* In NPTL, unlocking a mutex in the child process after a
8ae002
-   fork() is currently unsafe, whereas re-initializing it is safe and
8ae002
-   does not leak resources.  Therefore, a special atfork handler is
8ae002
-   installed for the child. */
8ae002
-
8ae002
-static void
8ae002
-ptmalloc_unlock_all2 (void)
8ae002
+void
8ae002
+internal_function
8ae002
+__malloc_fork_unlock_child (void)
8ae002
 {
8ae002
   mstate ar_ptr;
8ae002
 
8ae002
@@ -338,12 +330,6 @@ ptmalloc_unlock_all2 (void)
8ae002
   atfork_recursive_cntr = 0;
8ae002
 }
8ae002
 
8ae002
-# else
8ae002
-
8ae002
-#  define ptmalloc_unlock_all2 ptmalloc_unlock_all
8ae002
-
8ae002
-# endif
8ae002
-
8ae002
 #endif  /* !NO_THREADS */
8ae002
 
8ae002
 /* Initialization routine. */
8ae002
@@ -413,7 +399,6 @@ ptmalloc_init (void)
8ae002
 
8ae002
   tsd_key_create(&arena_key, NULL);
8ae002
   tsd_setspecific(arena_key, (void *)&main_arena);
8ae002
-  thread_atfork(ptmalloc_lock_all, ptmalloc_unlock_all, ptmalloc_unlock_all2);
8ae002
   const char *s = NULL;
8ae002
   if (__builtin_expect (_environ != NULL, 1))
8ae002
     {
8ae002
@@ -487,12 +472,6 @@ ptmalloc_init (void)
8ae002
   __malloc_initialized = 1;
8ae002
 }
8ae002
 
8ae002
-/* There are platforms (e.g. Hurd) with a link-time hook mechanism. */
8ae002
-#ifdef thread_atfork_static
8ae002
-thread_atfork_static(ptmalloc_lock_all, ptmalloc_unlock_all, \
8ae002
-		     ptmalloc_unlock_all2)
8ae002
-#endif
8ae002
-
8ae002
 
8ae002
 
8ae002
 /* Managing heaps and arenas (for concurrent threads) */
8ae002
@@ -827,7 +806,8 @@ _int_new_arena(size_t size)
8ae002
      limit is reached).  At this point, some arena has to be attached
8ae002
      to two threads.  We could acquire the arena lock before list_lock
8ae002
      to make it less likely that reused_arena picks this new arena,
8ae002
-     but this could result in a deadlock with ptmalloc_lock_all.  */
8ae002
+     but this could result in a deadlock with
8ae002
+     __malloc_fork_lock_parent.  */
8ae002
 
8ae002
   (void) mutex_lock (&a->mutex);
8ae002
 
8ae002
Index: b/malloc/malloc-internal.h
8ae002
===================================================================
8ae002
--- /dev/null
8ae002
+++ b/malloc/malloc-internal.h
8ae002
@@ -0,0 +1,32 @@
8ae002
+/* Internal declarations for malloc, for use within libc.
8ae002
+   Copyright (C) 2016 Free Software Foundation, Inc.
8ae002
+   This file is part of the GNU C Library.
8ae002
+
8ae002
+   The GNU C Library is free software; you can redistribute it and/or
8ae002
+   modify it under the terms of the GNU Lesser General Public License as
8ae002
+   published by the Free Software Foundation; either version 2.1 of the
8ae002
+   License, or (at your option) any later version.
8ae002
+
8ae002
+   The GNU C Library is distributed in the hope that it will be useful,
8ae002
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
8ae002
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
8ae002
+   Lesser General Public License for more details.
8ae002
+
8ae002
+   You should have received a copy of the GNU Lesser General Public
8ae002
+   License along with the GNU C Library; see the file COPYING.LIB.  If
8ae002
+   not, see <http://www.gnu.org/licenses/>.  */
8ae002
+
8ae002
+#ifndef _MALLOC_PRIVATE_H
8ae002
+#define _MALLOC_PRIVATE_H
8ae002
+
8ae002
+/* Called in the parent process before a fork.  */
8ae002
+void __malloc_fork_lock_parent (void) internal_function attribute_hidden;
8ae002
+
8ae002
+/* Called in the parent process after a fork.  */
8ae002
+void __malloc_fork_unlock_parent (void) internal_function attribute_hidden;
8ae002
+
8ae002
+/* Called in the child process after a fork.  */
8ae002
+void __malloc_fork_unlock_child (void) internal_function attribute_hidden;
8ae002
+
8ae002
+
8ae002
+#endif /* _MALLOC_PRIVATE_H */
8ae002
Index: b/malloc/malloc.c
8ae002
===================================================================
8ae002
--- a/malloc/malloc.c
8ae002
+++ b/malloc/malloc.c
8ae002
@@ -291,6 +291,7 @@ __malloc_assert (const char *assertion,
8ae002
 }
8ae002
 #endif
8ae002
 
8ae002
+#include <malloc/malloc-internal.h>
8ae002
 
8ae002
 /*
8ae002
   INTERNAL_SIZE_T is the word-size used for internal bookkeeping
8ae002
Index: b/malloc/tst-malloc-fork-deadlock.c
8ae002
===================================================================
8ae002
--- /dev/null
8ae002
+++ b/malloc/tst-malloc-fork-deadlock.c
8ae002
@@ -0,0 +1,220 @@
8ae002
+/* Test concurrent fork, getline, and fflush (NULL).
8ae002
+   Copyright (C) 2016 Free Software Foundation, Inc.
8ae002
+   This file is part of the GNU C Library.
8ae002
+
8ae002
+   The GNU C Library is free software; you can redistribute it and/or
8ae002
+   modify it under the terms of the GNU Lesser General Public License as
8ae002
+   published by the Free Software Foundation; either version 2.1 of the
8ae002
+   License, or (at your option) any later version.
8ae002
+
8ae002
+   The GNU C Library is distributed in the hope that it will be useful,
8ae002
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
8ae002
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
8ae002
+   Lesser General Public License for more details.
8ae002
+
8ae002
+   You should have received a copy of the GNU Lesser General Public
8ae002
+   License along with the GNU C Library; see the file COPYING.LIB.  If
8ae002
+   not, see <http://www.gnu.org/licenses/>.  */
8ae002
+
8ae002
+#include <sys/wait.h>
8ae002
+#include <unistd.h>
8ae002
+#include <errno.h>
8ae002
+#include <stdio.h>
8ae002
+#include <pthread.h>
8ae002
+#include <stdbool.h>
8ae002
+#include <stdlib.h>
8ae002
+#include <malloc.h>
8ae002
+#include <time.h>
8ae002
+#include <string.h>
8ae002
+#include <signal.h>
8ae002
+
8ae002
+static int do_test (void);
8ae002
+#define TEST_FUNCTION do_test ()
8ae002
+#include "../test-skeleton.c"
8ae002
+
8ae002
+enum {
8ae002
+  /* Number of threads which call fork.  */
8ae002
+  fork_thread_count = 4,
8ae002
+  /* Number of threads which call getline (and, indirectly,
8ae002
+     malloc).  */
8ae002
+  read_thread_count = 8,
8ae002
+};
8ae002
+
8ae002
+static bool termination_requested;
8ae002
+
8ae002
+static void *
8ae002
+fork_thread_function (void *closure)
8ae002
+{
8ae002
+  while (!__atomic_load_n (&termination_requested, __ATOMIC_RELAXED))
8ae002
+    {
8ae002
+      pid_t pid = fork ();
8ae002
+      if (pid < 0)
8ae002
+        {
8ae002
+          printf ("error: fork: %m\n");
8ae002
+          abort ();
8ae002
+        }
8ae002
+      else if (pid == 0)
8ae002
+        _exit (17);
8ae002
+
8ae002
+      int status;
8ae002
+      if (waitpid (pid, &status, 0) < 0)
8ae002
+        {
8ae002
+          printf ("error: waitpid: %m\n");
8ae002
+          abort ();
8ae002
+        }
8ae002
+      if (!WIFEXITED (status) || WEXITSTATUS (status) != 17)
8ae002
+        {
8ae002
+          printf ("error: waitpid returned invalid status: %d\n", status);
8ae002
+          abort ();
8ae002
+        }
8ae002
+    }
8ae002
+  return NULL;
8ae002
+}
8ae002
+
8ae002
+static char *file_to_read;
8ae002
+
8ae002
+static void *
8ae002
+read_thread_function (void *closure)
8ae002
+{
8ae002
+  FILE *f = fopen (file_to_read, "r");
8ae002
+  if (f == NULL)
8ae002
+    {
8ae002
+      printf ("error: fopen (%s): %m\n", file_to_read);
8ae002
+      abort ();
8ae002
+    }
8ae002
+
8ae002
+  while (!__atomic_load_n (&termination_requested, __ATOMIC_RELAXED))
8ae002
+    {
8ae002
+      rewind (f);
8ae002
+      char *line = NULL;
8ae002
+      size_t line_allocated = 0;
8ae002
+      ssize_t ret = getline (&line, &line_allocated, f);
8ae002
+      if (ret < 0)
8ae002
+        {
8ae002
+          printf ("error: getline: %m\n");
8ae002
+          abort ();
8ae002
+        }
8ae002
+      free (line);
8ae002
+    }
8ae002
+  fclose (f);
8ae002
+
8ae002
+  return NULL;
8ae002
+}
8ae002
+
8ae002
+static void *
8ae002
+flushall_thread_function (void *closure)
8ae002
+{
8ae002
+  while (!__atomic_load_n (&termination_requested, __ATOMIC_RELAXED))
8ae002
+    if (fflush (NULL) != 0)
8ae002
+      {
8ae002
+        printf ("error: fflush (NULL): %m\n");
8ae002
+        abort ();
8ae002
+      }
8ae002
+  return NULL;
8ae002
+}
8ae002
+
8ae002
+static void
8ae002
+create_threads (pthread_t *threads, size_t count, void *(*func) (void *))
8ae002
+{
8ae002
+  for (size_t i = 0; i < count; ++i)
8ae002
+    {
8ae002
+      int ret = pthread_create (threads + i, NULL, func, NULL);
8ae002
+      if (ret != 0)
8ae002
+        {
8ae002
+          errno = ret;
8ae002
+          printf ("error: pthread_create: %m\n");
8ae002
+          abort ();
8ae002
+        }
8ae002
+    }
8ae002
+}
8ae002
+
8ae002
+static void
8ae002
+join_threads (pthread_t *threads, size_t count)
8ae002
+{
8ae002
+  for (size_t i = 0; i < count; ++i)
8ae002
+    {
8ae002
+      int ret = pthread_join (threads[i], NULL);
8ae002
+      if (ret != 0)
8ae002
+        {
8ae002
+          errno = ret;
8ae002
+          printf ("error: pthread_join: %m\n");
8ae002
+          abort ();
8ae002
+        }
8ae002
+    }
8ae002
+}
8ae002
+
8ae002
+/* Create a file which consists of a single long line, and assigns
8ae002
+   file_to_read.  The hope is that this triggers an allocation in
8ae002
+   getline which needs a lock.  */
8ae002
+static void
8ae002
+create_file_with_large_line (void)
8ae002
+{
8ae002
+  int fd = create_temp_file ("bug19431-large-line", &file_to_read);
8ae002
+  if (fd < 0)
8ae002
+    {
8ae002
+      printf ("error: create_temp_file: %m\n");
8ae002
+      abort ();
8ae002
+    }
8ae002
+  FILE *f = fdopen (fd, "w+");
8ae002
+  if (f == NULL)
8ae002
+    {
8ae002
+      printf ("error: fdopen: %m\n");
8ae002
+      abort ();
8ae002
+    }
8ae002
+  for (int i = 0; i < 50000; ++i)
8ae002
+    fputc ('x', f);
8ae002
+  fputc ('\n', f);
8ae002
+  if (ferror (f))
8ae002
+    {
8ae002
+      printf ("error: fputc: %m\n");
8ae002
+      abort ();
8ae002
+    }
8ae002
+  if (fclose (f) != 0)
8ae002
+    {
8ae002
+      printf ("error: fclose: %m\n");
8ae002
+      abort ();
8ae002
+    }
8ae002
+}
8ae002
+
8ae002
+static int
8ae002
+do_test (void)
8ae002
+{
8ae002
+  /* Make sure that we do not exceed the arena limit with the number
8ae002
+     of threads we configured.  */
8ae002
+  if (mallopt (M_ARENA_MAX, 400) == 0)
8ae002
+    {
8ae002
+      printf ("error: mallopt (M_ARENA_MAX) failed\n");
8ae002
+      return 1;
8ae002
+    }
8ae002
+
8ae002
+  /* Leave some room for shutting down all threads gracefully.  */
8ae002
+  int timeout = 3;
8ae002
+  if (timeout > TIMEOUT)
8ae002
+    timeout = TIMEOUT - 1;
8ae002
+
8ae002
+  create_file_with_large_line ();
8ae002
+
8ae002
+  pthread_t fork_threads[fork_thread_count];
8ae002
+  create_threads (fork_threads, fork_thread_count, fork_thread_function);
8ae002
+  pthread_t read_threads[read_thread_count];
8ae002
+  create_threads (read_threads, read_thread_count, read_thread_function);
8ae002
+  pthread_t flushall_threads[1];
8ae002
+  create_threads (flushall_threads, 1, flushall_thread_function);
8ae002
+
8ae002
+  struct timespec ts = {timeout, 0};
8ae002
+  if (nanosleep (&ts, NULL))
8ae002
+    {
8ae002
+      printf ("error: error: nanosleep: %m\n");
8ae002
+      abort ();
8ae002
+    }
8ae002
+
8ae002
+  __atomic_store_n (&termination_requested, true, __ATOMIC_RELAXED);
8ae002
+
8ae002
+  join_threads (flushall_threads, 1);
8ae002
+  join_threads (read_threads, read_thread_count);
8ae002
+  join_threads (fork_threads, fork_thread_count);
8ae002
+
8ae002
+  free (file_to_read);
8ae002
+
8ae002
+  return 0;
8ae002
+}
8ae002
Index: b/manual/memory.texi
8ae002
===================================================================
8ae002
--- a/manual/memory.texi
8ae002
+++ b/manual/memory.texi
8ae002
@@ -1055,14 +1055,6 @@ systems that do not support @w{ISO C11}.
8ae002
 @c     _dl_addr_inside_object ok
8ae002
 @c    determine_info ok
8ae002
 @c    __rtld_lock_unlock_recursive (dl_load_lock) @aculock
8ae002
-@c   thread_atfork @asulock @aculock @acsfd @acsmem
8ae002
-@c    __register_atfork @asulock @aculock @acsfd @acsmem
8ae002
-@c     lll_lock (__fork_lock) @asulock @aculock
8ae002
-@c     fork_handler_alloc @asulock @aculock @acsfd @acsmem
8ae002
-@c      calloc dup @asulock @aculock @acsfd @acsmem
8ae002
-@c     __linkin_atfork ok
8ae002
-@c      catomic_compare_and_exchange_bool_acq ok
8ae002
-@c     lll_unlock (__fork_lock) @aculock
8ae002
 @c   *_environ @mtsenv
8ae002
 @c   next_env_entry ok
8ae002
 @c   strcspn dup ok
8ae002
Index: b/nptl/sysdeps/unix/sysv/linux/fork.c
8ae002
===================================================================
8ae002
--- a/nptl/sysdeps/unix/sysv/linux/fork.c
8ae002
+++ b/nptl/sysdeps/unix/sysv/linux/fork.c
8ae002
@@ -29,7 +29,7 @@
8ae002
 #include <bits/stdio-lock.h>
8ae002
 #include <atomic.h>
8ae002
 #include <pthreadP.h>
8ae002
-
8ae002
+#include <malloc/malloc-internal.h>
8ae002
 
8ae002
 unsigned long int *__fork_generation_pointer;
8ae002
 
8ae002
@@ -116,6 +116,11 @@ __libc_fork (void)
8ae002
 
8ae002
   _IO_list_lock ();
8ae002
 
8ae002
+  /* Acquire malloc locks.  This needs to come last because fork
8ae002
+     handlers may use malloc, and the libio list lock has an indirect
8ae002
+     malloc dependency as well (via the getdelim function).  */
8ae002
+  __malloc_fork_lock_parent ();
8ae002
+
8ae002
 #ifndef NDEBUG
8ae002
   pid_t ppid = THREAD_GETMEM (THREAD_SELF, tid);
8ae002
 #endif
8ae002
@@ -172,6 +177,9 @@ __libc_fork (void)
8ae002
 # endif
8ae002
 #endif
8ae002
 
8ae002
+      /* Release malloc locks.  */
8ae002
+      __malloc_fork_unlock_child ();
8ae002
+
8ae002
       /* Reset the file list.  These are recursive mutexes.  */
8ae002
       fresetlockfiles ();
8ae002
 
8ae002
@@ -213,6 +221,9 @@ __libc_fork (void)
8ae002
       /* Restore the PID value.  */
8ae002
       THREAD_SETMEM (THREAD_SELF, pid, parentpid);
8ae002
 
8ae002
+      /* Release malloc locks, parent process variant.  */
8ae002
+      __malloc_fork_unlock_parent ();
8ae002
+
8ae002
       /* We execute this even if the 'fork' call failed.  */
8ae002
       _IO_list_unlock ();
8ae002