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 :

Pointeurs et bus error


Sujet :

C

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre averti
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    33
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2006
    Messages : 33
    Par défaut Pointeurs et bus error
    Bonjour,

    Etudiant en info, je travaille sur une fonction qui doit inverser deux a deux les caracteres d'une chaine, a savoir la premiere lettre avec la derniere, la seconde avec l'avant derniere, etc ...

    Probleme : Ayant retourne le probleme dans TOUS les sens, je prends des "Bus error" sur 'Bus error". Parfois il m'est peut-etre arrive de prendre un ou deux segfault, dans l'idee je suis bloque donc je me permets de faire appel a votre aide

    Voici le code qui Bus error :

    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
    void my_putchar(char c)
    {
    write(1, &c, 1);
    }
    
    char  *my_evil_string(char *str)
    {
    int tabSize = my_strlen(str);
    int nbInversions = tabSize / 2;
    int count = 0;
    char adresseTmp;
    while(count < nbInversions)
    {
    adresseTmp = str[count];
    str[count] = str[tabSize - count - 1];
    str[tabSize - count - 1 = adresseTmp];
    count++;
    }
    }
    
    void main(0)
    {
    my_evil_str("ABCDEFGHIJKLMNOPQRSTUVWXYZ");
    }
    Je garantis pas que le code compile, juste pour donner une idee de mon probleme.

    En gras, les lignes qui font un Bus Error.

    D'ou vient ce probleme ?

    Grand MERCI d'avance

  2. #2
    Membre émérite Avatar de SofEvans
    Homme Profil pro
    Développeur C
    Inscrit en
    Mars 2009
    Messages
    1 084
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 36
    Localisation : France

    Informations professionnelles :
    Activité : Développeur C

    Informations forums :
    Inscription : Mars 2009
    Messages : 1 084
    Par défaut
    Bonjour,

    Et ben c'est le grand bazar dans ton code.
    Encore et toujours la meme liste de conseil.


    **
    Fonction main non conforme !

    Le standart est tres clair dessus :

    int main ()

    ou

    int main (int argc, char *argv[])


    **
    La fonction utilisé dans la main ainsi que celle definie n'ont pas les meme nom !
    ""my_evil_str" vs "my_evil_string"


    **
    str[tabSize - count - 1 = adresseTmp];

    waouh, c'est possible ca ? Je crois que tu as dû inverser ...
    str[adresseTmp = (tabSize - count - 1)];


    **
    my_evil_string retourne un char*
    Or, tu ne recupere pas cette valeur dans l'appelant et tu ne fais pas de return !
    Si my_evil_string doit retourner un char*, quel est-il ??


    **
    write(1, &c, 1);

    Pour plus de rigueur,


    write(1, &c, sizeof(char));

    car le troisieme parametre est de type size.

    **
    Je n'ai pas la declaration de my_strlen, je te fais donc confiance la dessus, j'utilise strlen.


    **
    Ensuite, une bourde classique :
    Tu utilise un while au lieu d'un for.
    Pourquoi utiliser un for ?
    Parce que tu sais a tout les coup combien de fois tu va devoir inverser les caractere ! C'est donc une boucle deterministe et non indeterministe comme while.


    **
    Je te proposerai bientot une correction.



    EDIT :
    J'ai une petite idée quant on ton erreur.
    Lorsque tu passe ta chaine dans my_evil_string(char str[]), la chaine est automatiquement completer par NULL ayant pour code ascii 0.
    Si dans ta fonction my_strlen tu compte en fonction de '\n', tu va faire un buffer overflow, autrment un depassement de ton tableau.
    Pourquoi ?
    Parce que tu vas continuer a regarder a la suite du tableau jusqu'a ce que tu trouve '\n', ce qui peut durer longtemps.
    En resumer, tu sort des limite de ton tableau.

  3. #3
    Membre émérite Avatar de SofEvans
    Homme Profil pro
    Développeur C
    Inscrit en
    Mars 2009
    Messages
    1 084
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 36
    Localisation : France

    Informations professionnelles :
    Activité : Développeur C

    Informations forums :
    Inscription : Mars 2009
    Messages : 1 084
    Par défaut
    Voici ma reponse.
    Malheuresement, je n'ai pas reussi a faire ce que je voulais.
    J'obtenais moi aussi un SIGSEV sur mon tableau en ayant les bon indice.


    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
    40
    41
    42
    43
     
    #include <stdio.h>
    #include <stdlib.h>
     
    #include <unistd.h>
     
     
    int  my_strlen(char str[])
    {
        int nombreDeLettre = 0;
        while (str[nombreDeLettre] != '\0')
            nombreDeLettre++;
     
        return nombreDeLettre;
    }
     
    void my_putchar(char c)
    {
        write(1, &c, sizeof(char));
    }
     
    void my_evil_string(char str[])
    {
        int nombreDeLettre = my_strlen(str);
     
        int i;
        for (i=0 ; i<nombreDeLettre/2 ; i++)
        {
            char temporaire = str[i];
            str[i] = str[nombreDeLettre - 1 - i];
            str[nombreDeLettre - 1 - i] = temporaire;
        }
    }
     
    int main (int argc, char *argv[])
    {
        char str[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
        my_evil_string(str);
     
        printf("%s\n",str);
     
        return EXIT_SUCCESS;
    }

    Le probleme venait d'ici :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
        my_evil_string("ABCDEFGHIJKLMNOPQRSTUVWXYZ");
    En passant la chaine de cette maniere, il y avait sigsev.
    J'ai donc dû faire ceci :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
     
        char str[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
        my_evil_string(str);
    La question maintenant est de savoir pourquoi il y a un sigsev, et comment faire en sorte de l'éviter tout en conservant la première écriture.

    Des idées ?

  4. #4
    Membre averti
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    33
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2006
    Messages : 33
    Par défaut
    Salut et merci pour ta reponse precise ^^

    Desole pour les quelques erreurs mais n'ayant pas fait de copier coller du code, il contient forcement quelques erreurs idiotes ( genre le adresseTmp - count - 1 qui a vu un egal remplacer un moins ^^)

    Bref merci pour ta correction, par contre j'aimerais sincerement comprendre pourquoi on se prend ces bus error. J'ai tout teste : rien a faire =(

    Encore merci a toi

  5. #5
    Expert confirmé
    Avatar de diogene
    Homme Profil pro
    Enseignant Chercheur
    Inscrit en
    Juin 2005
    Messages
    5 761
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Enseignant Chercheur
    Secteur : Enseignement

    Informations forums :
    Inscription : Juin 2005
    Messages : 5 761
    Par défaut
    SofEvans :
    Le probleme venait d'ici :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
        my_evil_string("ABCDEFGHIJKLMNOPQRSTUVWXYZ");
    En passant la chaine de cette maniere, il y avait sigsev.
    J'ai donc dû faire ceci :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
        char str[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
        my_evil_string(str);
    La question maintenant est de savoir pourquoi il y a un sigsev, et comment faire en sorte de l'éviter tout en conservant la première écriture.
    La chaine "ABCDEFGHIJKLMNOPQRSTUVWXYZ", dans le premier cas est non modifiable : elle est placée dans la zone mémoire attribuée au code du programme, pas dans celles attribuées aux variables du programme. Or my_evil_string() essaye de modifier cette chaine.
    C'est également le cas ici :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
        char *str = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
        my_evil_string(str);
    // donc on doit écrire par précaution 
    // char const *str = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
    Dans le code utilisant un tableau, la situation est différente puisqu'il s'agit de l'initialisation du tableau qui lui est modifiable.

  6. #6
    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
    Si tu compiles avec gcc, ajoute l'option -Wwrite-strings, qui te signalera ce genre d'erreur.
    Si tu veux que ta fonction puisse faire des calculs modifiants sur une chaîne alors que la chaîne reçue est constante, il te faudra dupliquer la chaîne, possiblement par allocation dynamique (malloc() etc.).

    PS: L'année vient de commencer à EPITA/EPITECH? C'est en quelque jours le troisième (au moins) thread de manipulation de chaînes par quelqu'un qui ne peut pas utiliser les IO standard (printf etc.) et doit utiliser POSIX (write() etc.) à la place...
    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.

  7. #7
    Membre émérite Avatar de SofEvans
    Homme Profil pro
    Développeur C
    Inscrit en
    Mars 2009
    Messages
    1 084
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 36
    Localisation : France

    Informations professionnelles :
    Activité : Développeur C

    Informations forums :
    Inscription : Mars 2009
    Messages : 1 084
    Par défaut
    Citation Envoyé par diogene Voir le message
    SofEvans :


    La chaine "ABCDEFGHIJKLMNOPQRSTUVWXYZ", dans le premier cas est non modifiable : elle est placée dans la zone mémoire attribuée au code du programme, pas dans celles attribuées aux variables du programme. Or my_evil_string() essaye de modifier cette chaine.
    C'est également le cas ici :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
        char *str = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
        my_evil_string(str);
    // donc on doit écrire par précaution 
    // char const *str = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
    Dans le code utilisant un tableau, la situation est différente puisqu'il s'agit de l'initialisation du tableau qui lui est modifiable.
    Merci de ces information diogene !

    Cependant quelque chose me gene :

    La chaine "ABCDEFGHIJKLMNOPQRSTUVWXYZ", dans le premier cas est non modifiable ...
    C'est également le cas ici :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
        char *str = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
        my_evil_string(str);
    Je ne comprend pas. Dans le deuxieme cas, str est bien modifiable, non ?
    Pourquoi faire un const si justement on veut modifier la chaine ?

    Existe t'il une maniere de placer cette zone mémoire attribuée au code du programme dans la zone memoire attribué aux variables ?

Discussions similaires

  1. bus error bizzare
    Par markotik dans le forum C++
    Réponses: 3
    Dernier message: 20/01/2007, 19h58
  2. Probleme de valeurs fixes et bus error
    Par toto08 dans le forum C
    Réponses: 18
    Dernier message: 02/01/2007, 14h44
  3. [Débutant] Bus Error et scanf
    Par BiLLKiLL dans le forum C
    Réponses: 2
    Dernier message: 17/09/2006, 20h47
  4. bus error sur une machine SUN
    Par mhtrinh dans le forum C
    Réponses: 14
    Dernier message: 10/07/2006, 16h43
  5. bus error
    Par salseropom dans le forum C
    Réponses: 3
    Dernier message: 15/12/2005, 11h59

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