IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

C Discussion :

Problème waitpid attend pas


Sujet :

C

  1. #1
    Membre averti
    Inscrit en
    Octobre 2010
    Messages
    20
    Détails du profil
    Informations forums :
    Inscription : Octobre 2010
    Messages : 20
    Par défaut Problème waitpid attend pas
    Salut à tous !
    voilà j'ai un programme qui créé des processus fils, mais malheureusement, le waitpid ne marche que de temps en temps, ca dépend des jours.
    j'ai utilisé psignal, et mon fils est terminé par Unknown signal 0
    je vous montre le code et son resultat :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
     
    void attend(int n, int *etat, int *fpid)
    {
    	int i;
    	int status;
    	for(i=0;i<n;i++)
    	{
    		waitpid(fpid[i],&status,1);//attend pour tous les fils
    		psignal(status,"termination par ");
    		printf("errno = '%X'\n",errno);
    		if(WIFEXITED(status))
    			printf("terminé normalement\n");
    		if(WIFSIGNALED(status))
    		{
    			printf("signal non recepte\n");
    			printf("No signal = %d\n",WTERMSIG(status));
    		}
     
    	}
    }
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
     
    [miloutch@ubuntu:~/Bureau/shell2]$firefox
    termination*par*:*Unknown*signal*0
    errno*=*'0'
    terminé*normalement
    [miloutch@ubuntu:~/Bureau/shell2]$
    et bien entendu, le programme lancé ne s'est pas terminé !

    Comme je vou l'ai dit, j'ai remarqué que le waitpid attend mon fils que de temps en temps.
    un jour ca marche, un jour ca marche pas. Je n'ai aucune idée d'ou ca peut venir.
    en tout cas même si ca dépend des jours, ca marche, donc je suppose que l'erreur ne vient pas du code que j'ai fait pour le waitpid.
    si quelqu'un a déjà eu ce genre de problèmes, je suis preneur pour une solution.
    merci

  2. #2
    Membre Expert
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Octobre 2008
    Messages
    1 515
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 46
    Localisation : France

    Informations professionnelles :
    Activité : Ingénieur développement logiciels

    Informations forums :
    Inscription : Octobre 2008
    Messages : 1 515
    Par défaut
    Tu n'utilises pas waitpid() correctement. Et psignal() non plus d'ailleurs.

    Le dernier argument de waitpid() est un champs de bits qui indique des options. Tu n'es pas sensé passer un nombre sous forme littérale, mais plutôt utiliser les macros définies dans wait.h : WNOHANG, WUNTRACED, WCONTINUED... Si tu passes 1, alors c'est comme si tu avais passé WNOHANG, qui demande justement à waitpid() de ne pas attendre... Ce qui n'est pas ce que tu veux. Dans ton cas il faut probablement faire waitpid(fpid[i], &status, 0).

    Le premier argument de psignal() est un numéro de signal. Toi tu lui passes le status retourné par waitpid(), qui n'est pas une numéro de signal. Il est donc normal d'avoir "Unknown signal 0". Le numéro de signal est WTERMSIG(status), à n'utiliser que si WIFSIGNALED(status) est vrai.

  3. #3
    Membre averti
    Inscrit en
    Octobre 2010
    Messages
    20
    Détails du profil
    Informations forums :
    Inscription : Octobre 2010
    Messages : 20
    Par défaut
    j'ai modifié mon programme comme tu l'as dit, et du coup le processus a l'air de se couper normalement, sauf que c'est pas normal
    du coup il ne m'affiche pas les erreurs signal WTERMSIG.
    le 0 à la place de waitpid ne change rien malheureusement
    merci quand même pour la réponse

  4. #4
    Membre Expert
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Octobre 2008
    Messages
    1 515
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 46
    Localisation : France

    Informations professionnelles :
    Activité : Ingénieur développement logiciels

    Informations forums :
    Inscription : Octobre 2008
    Messages : 1 515
    Par défaut
    Si tu étais plus clair ça donnerait peut-être envie de te répondre... Parce que "le processus a l'air de se couper normalement, sauf que c'est pas normal", moi, je ne comprend pas.

    Le plus simple c'est que tu donnes ton code actuel.

  5. #5
    Membre averti
    Inscrit en
    Octobre 2010
    Messages
    20
    Détails du profil
    Informations forums :
    Inscription : Octobre 2010
    Messages : 20
    Par défaut
    Non mais franchement ca vient pas du code, c'est pas possible, puisque le wait ne peut pas ne pas marcher , puis tout d'un coup fonctionner. ca doit venir de l'environnement. bref je me debrouillerai. je mets le code à tous hasard :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
     
    do
    {
    	if(c[j].pipe!=0)//si il y a pipe
    	{
    		if(pipe(fpipe) == -1)//on le crée
    		{
    			perror("erreur lors de la création du pipe");
    			exit(0);
    		}
    		c[j].sortie=fpipe[1] ;//redirige la sortie vers le pipe
    	}
    	if ((fpid[j] = fork()) < 0)//crée le processus fils
    	{
    		perror("erreur lors de la création d'un fils");
    		exit(0);
    	}
    	//processus fils
    	if(fpid[j]==0)
    	{
    		redirection(c+j);//vérifie les redirections et les mets en place
    		if(c[j].pipe!=0)//si il y avait pipe
    			close(fpipe[0]);//on ferme la sortie
    		execute(c[j]);//execute la commande
    	}
    	if(c[j].pipe!=0)//si il y avait pipe
    	{  
    		close(fpipe[1]);//on ferme l'entrée
    		c[j+1].entree=fpipe[0];//redirige l'entrée de la commande suivante vers le pipe  
    	}
    	c[j+1].sortie=-1;//on coupe la sortie de la prochaine commande
    	j++;//commande suivante
    }while(c[j-1].pipe);//on parcours toutes les commandes (un | crée une nouvelle commande)
     
    //si en mode background, on attend pas , sinon on attend
    if(bg!=1)
    {
    	attend(j,NULL,fpid);
    }
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
     
    //attend le(s) processus fils en cours, n =  nombre de pipe
    void attend(int n, int *etat, int *fpid)
    {
    	int i;
    	int status;
    	for(i=0;i<n;i++)
    	{
    		waitpid(fpid[i],&status,0);//attend pour tous les fils
    		psignal(WTERMSIG(status),"termination par ");
    		printf("errno = '%X'\n",errno);
    		if(WIFEXITED(status))
    			printf("terminé normalement\n");
    		if(WIFSIGNALED(status))
    		{
    			printf("signal non recepte\n");
    			printf("No signal = %d\n",WTERMSIG(status));
    		}
     
    	}
    }

  6. #6
    Membre Expert Avatar de jabbounet
    Homme Profil pro
    Consultant informatique
    Inscrit en
    Juin 2009
    Messages
    1 909
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 49

    Informations professionnelles :
    Activité : Consultant informatique

    Informations forums :
    Inscription : Juin 2009
    Messages : 1 909
    Par défaut
    Non mais franchement ca vient pas du code, c'est pas possible, puisque le wait ne peut pas ne pas marcher , puis tout d'un coup fonctionner. ca doit venir de l'environnement. bref je me debrouillerai. je mets le code à tous hasard :
    Ces fonctions existent depuis les débuts d'unix, elles sont aussi utilisé par tout un tas d'applications/services de ton os, si elles ne marchaient pas ton os serait probablement instable.
    Quand elles sont utilisées correctement elles ne posent pas de problèmes.



    Dans "attend" tu ne vérifie pas que tu es dans le processus père quand tu appelle wait, ce qui fait que les fils aussi appellent wait.
    Et comme fork retourne 0 quand tu es le processus fils, l'appel de waipid dans ce cas attendra la fin d'un fils du fils ....

  7. #7
    Membre Expert Avatar de jabbounet
    Homme Profil pro
    Consultant informatique
    Inscrit en
    Juin 2009
    Messages
    1 909
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 49

    Informations professionnelles :
    Activité : Consultant informatique

    Informations forums :
    Inscription : Juin 2009
    Messages : 1 909
    Par défaut
    personnellement je partirai sur une solution dans ce style dans le cas d'un père générant plusieurs fils qu'il doit ensuite attendre.

    Ce code est repris et adapté d'un vieux post de ce forum, plus des exemples de man page trouvé sur le net. je n'ai pas de compilateur sous la main, mais le principe est là.
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
     
    int i;
    int isFils = 0;
    pid_t tabPid[NB_PID];
    for (i=0; i < NB_PID; i++)
    {
        switch(tabPid[i]=fork())
        {
              case -1: /* Erreur de fork - Cas à traiter */
                    perror("fork"); exit(EXIT_FAILURE);
              case 0: // Fils
                    fils(i); // appel de la fonction fils
                    isFils = 1;  
                    break;
              default: 
                    printf("fils %d crée avec le pid %d\n",i,tabPid[i]);
                    break;
        }
    }
     
    if (isFils == 0) { /* nous sommes dans le père, avec mode background */
       int status;
       for(i = 0; i < NB_PID ; i ++) {
    	 pid_t w = waitpid(tabPid[i], &status, 0);
             if (w == -1) { perror("waitpid"); exit(EXIT_FAILURE); }
             if (WIFEXITED(status)) {
                    printf("exited, status=%d\n", WEXITSTATUS(status));
             } else if (WIFSIGNALED(status)) {
                    printf("killed by signal %d\n", WTERMSIG(status));
             } else if (WIFSTOPPED(status)) {
                    printf("stopped by signal %d\n", WSTOPSIG(status));
             } 
       }
    }
    je n'ai pas repris ton code avec les pipes pour ne pas alourdir la partie fork/waitpid

  8. #8
    Membre averti
    Inscrit en
    Octobre 2010
    Messages
    20
    Détails du profil
    Informations forums :
    Inscription : Octobre 2010
    Messages : 20
    Par défaut
    merci beaucoup pour le code, je regarde ca demain

  9. #9
    Membre averti
    Inscrit en
    Octobre 2010
    Messages
    20
    Détails du profil
    Informations forums :
    Inscription : Octobre 2010
    Messages : 20
    Par défaut
    gros boulet que je suis !
    bon bah j'ai trouvé la source du problème lol.
    en fait, lorsque une instance de firefox est déjà ouverte lorsque je tape la commande, ca lance une nouvelle fenetre firefox et le wait n'attend pas. si il n'y en a pas, il lance firefox et attend que je le coupe pour continuer.
    je crois que c'est normal, non ?

  10. #10
    Expert éminent
    Avatar de Médinoc
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Septembre 2005
    Messages
    27 395
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 41
    Localisation : France

    Informations professionnelles :
    Activité : Développeur informatique
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Septembre 2005
    Messages : 27 395
    Par défaut
    oui.
    SVP, pas de questions techniques par MP. Surtout si je ne vous ai jamais parlé avant.

    "Aw, come on, who would be so stupid as to insert a cast to make an error go away without actually fixing the error?"
    Apparently everyone.
    -- Raymond Chen.
    Traduction obligatoire: "Oh, voyons, qui serait assez stupide pour mettre un cast pour faire disparaitre un message d'erreur sans vraiment corriger l'erreur?" - Apparemment, tout le monde. -- Raymond Chen.

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. [AJAX] settimeout qui attend pas assez
    Par zooffy dans le forum Général JavaScript
    Réponses: 7
    Dernier message: 11/01/2007, 10h52
  2. Réponses: 1
    Dernier message: 14/12/2006, 22h47
  3. Problème de microphone pas de détéction
    Par rub091 dans le forum Windows XP
    Réponses: 6
    Dernier message: 03/08/2006, 16h38
  4. Problème BDE ou pas ?
    Par Jeff on the web dans le forum Bases de données
    Réponses: 7
    Dernier message: 12/07/2006, 09h22
  5. [Mail] Problème fonction mail() pas de sujet ni corps!
    Par webrider dans le forum Langage
    Réponses: 3
    Dernier message: 28/06/2006, 17h04

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo