/* Exemple d'illustration des primitives Unix : Un père et ses fils */ /* SIGALRM et waitpid pour prendre connaissance de la fin des fils */ #include /* entrées sorties */ #include /* pimitives de base : fork, ...*/ #include /* exit */ #include /* wait */ #include /* opérations sur les signaux */ #define NB_FILS 3 /* nombre de fils */ #define D_ALARM 10 /* durée pour alarm */ int nb_fils_termines; /* variable globale car modifiée par le traitant */ /* Traitant du signal SIGCHLD */ void handler_sigalrm(int signal_num) { int fils_termine, wstatus; if (signal_num == SIGALRM) { while ((fils_termine = (int) waitpid(-1, &wstatus, WNOHANG | WUNTRACED | WCONTINUED)) > 0) { if WIFEXITED(wstatus) { printf("\nMon fils de pid %d s'est arrete avec exit %d\n", fils_termine, WEXITSTATUS(wstatus)); nb_fils_termines++; } else if WIFSIGNALED(wstatus) { printf("\nMon fils de pid %d a recu le signal %d\n", fils_termine, WTERMSIG(wstatus)); nb_fils_termines++; } else if (WIFCONTINUED(wstatus)) { printf("\nMon fils de pid %d a ete relance \n", fils_termine); } else if (WIFSTOPPED(wstatus)) { printf("\nMon fils de pid %d a ete suspendu \n", fils_termine); } } } /* relancer alarm a cela n'est pas fait automatiquement */ alarm(D_ALARM); return; } int main() { int fils, retour; int duree_sommeil = 3; nb_fils_termines = 0; /* associer traitant sigchld à SIGCHLD */ signal(SIGALRM, handler_sigalrm); printf("\nJe suis le processus principal de pid %d\n", getpid()); /* Vidange du tampon de sortie pour que le fils le récupère vide */ fflush(stdout); for (fils = 1; fils <= NB_FILS; fils++) { retour = fork(); /* Bonne pratique : tester systématiquement le retour des appels système */ if (retour < 0) { /* échec du fork */ printf("Erreur fork\n"); /* Convention : s'arrêter avec une valeur > 0 en cas d'erreur */ exit(1); } /* fils */ if (retour == 0) { printf("\n Processus fils numero %d, de pid %d, de pere %d.\n", fils, getpid(), getppid()); sleep(duree_sommeil * fils); printf("\n Processus fils numero %d : je m\'arrête.\n", fils); exit(fils); /* normalement exit(0), mais on veut illustrer WEXITSTATUS */ } /* pere */ else { printf("\nProcessus de pid %d a cree un fils numero %d, de pid %d \n", getpid(), fils, retour); } } alarm(D_ALARM); /* faire ce qu'on veut jusqu'à la terminaison de tous les fils */ do { /* faire ce qu'on veut */ } while (nb_fils_termines < NB_FILS); printf("\nProcessus Principal termine\n"); return EXIT_SUCCESS; }