| /* Tests for fork in multi-threaded environment. |
| Copyright (C) 2000 Free Software Foundation, Inc. |
| Contributed by Ulrich Drepper <drepper@cygnus.com>, 2000. |
| |
| The GNU C Library 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. |
| |
| The GNU C Library 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. |
| |
| You should have received a copy of the GNU Library General Public |
| License along with the GNU C Library; see the file COPYING.LIB. If not, |
| write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, |
| Boston, MA 02111-1307, USA. */ |
| |
| #include <errno.h> |
| #if !defined(__APPLE__) |
| # include <error.h> |
| #endif |
| #include <stdlib.h> |
| #include <pthread.h> |
| #include <unistd.h> |
| #include <sys/wait.h> |
| #include <stdio.h> |
| |
| #if defined(__APPLE__) |
| #include <string.h> /* strerror */ |
| static void error (int status, int errnum, char* msg) |
| { |
| fprintf(stderr, "%s%s%s\n", |
| msg, |
| errnum ? ": " : "", |
| errnum ? strerror(errnum) : ""); |
| if (errnum) |
| exit(errnum); |
| } |
| #endif |
| |
| enum |
| { |
| PREPARE_BIT = 1, |
| PARENT_BIT = 2, |
| CHILD_BIT = 4 |
| }; |
| |
| static int var; |
| |
| static void |
| prepare (void) |
| { |
| var |= PREPARE_BIT; |
| } |
| |
| static void |
| parent (void) |
| { |
| var |= PARENT_BIT; |
| } |
| |
| static void |
| child (void) |
| { |
| var |= CHILD_BIT; |
| } |
| |
| |
| static void *thread (void *arg); |
| |
| |
| int |
| main (void) |
| { |
| pthread_t th; |
| void *res; |
| |
| pthread_atfork (prepare, parent, child); |
| |
| if (pthread_create (&th, NULL, thread, NULL) != 0) |
| error (EXIT_FAILURE, 0, "cannot create thread"); |
| |
| pthread_join (th, &res); |
| |
| if ( ( int ) ( long int ) res != 0 ) |
| error(EXIT_FAILURE, 0, "pthread_join res != 0" ); |
| |
| printf ( "all ok\n" ); |
| return 0; |
| } |
| |
| |
| static void * |
| thread (void *arg) |
| { |
| int status; |
| pid_t pid; |
| |
| pid = fork (); |
| if (pid == 0) |
| { |
| /* We check whether the `prepare' and `child' function ran. */ |
| exit (var != (PREPARE_BIT | CHILD_BIT)); |
| } |
| else if (pid == (pid_t) -1) |
| error (EXIT_FAILURE, errno, "cannot fork"); |
| |
| if (waitpid (pid, &status, 0) != pid) |
| error (EXIT_FAILURE, errno, "wrong child"); |
| |
| if (WTERMSIG (status) != 0) |
| error (EXIT_FAILURE, 0, "Child terminated incorrectly"); |
| status = WEXITSTATUS (status); |
| |
| if (status == 0) |
| status = var != (PREPARE_BIT | PARENT_BIT); |
| |
| return (void *) (long int) status; |
| } |