| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293 | /* Linuxthreads - a simple clone()-based implementation of Posix        *//* threads for Linux.                                                   *//* Copyright (C) 1996 Xavier Leroy (Xavier.Leroy@inria.fr)              *//*                                                                      *//* This program is free software; you can redistribute it and/or        *//* modify it under the terms of the GNU Library General Public License  *//* as published by the Free Software Foundation; either version 2       *//* of the License, or (at your option) any later version.               *//*                                                                      *//* This program is distributed in the hope that it will be useful,      *//* but WITHOUT ANY WARRANTY; without even the implied warranty of       *//* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the        *//* GNU Library General Public License for more details.                 *//* The "atfork" stuff */#include <errno.h>#include <stddef.h>#include <stdlib.h>#include <unistd.h>#include "pthread.h"#include "internals.h"#include <bits/libc-lock.h>#include "fork.h"extern int __libc_fork (void);pid_t __pthread_fork (struct fork_block *b){  pid_t pid;  list_t *runp;  __libc_lock_lock (b->lock);  /* Run all the registered preparation handlers.  In reverse order.  */  list_for_each_prev (runp, &b->prepare_list)    {      struct fork_handler *curp;      curp = list_entry (runp, struct fork_handler, list);      curp->handler ();    }  __pthread_once_fork_prepare();  __flockfilelist();  pid = ARCH_FORK ();  if (pid == 0) {    __pthread_reset_main_thread();    __fresetlockfiles();    __pthread_once_fork_child();    /* Run the handlers registered for the child.  */    list_for_each (runp, &b->child_list)      {	struct fork_handler *curp;	curp = list_entry (runp, struct fork_handler, list);	curp->handler ();      }    __libc_lock_init (b->lock);  } else {    __funlockfilelist();    __pthread_once_fork_parent();    /* Run the handlers registered for the parent.  */    list_for_each (runp, &b->parent_list)      {	struct fork_handler *curp;	curp = list_entry (runp, struct fork_handler, list);	curp->handler ();      }    __libc_lock_unlock (b->lock);  }  return pid;}#ifdef SHAREDpid_t __fork (void){  return __libc_fork ();}weak_alias (__fork, fork);pid_t __vfork(void){  return __libc_fork ();}weak_alias (__vfork, vfork);#endif
 |