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 :

Lecture sur port série sous Linux


Sujet :

C

  1. #1
    Nouveau membre du Club
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    39
    Détails du profil
    Informations personnelles :
    Âge : 44
    Localisation : France

    Informations forums :
    Inscription : Octobre 2006
    Messages : 39
    Points : 32
    Points
    32
    Par défaut Lecture sur port série sous Linux
    Bonjour.

    J'ai besoin de programmer l'accès à un port série en R/W en C sous Linux. J'ai actuellement réussi à écrire vers un terminal externe mais je n'arrive pas à lire des caractères envoyés depuis ce terminal.

    J'ai essayé de lire les codes dans les différentes discussions sur le prot RS232 (recherche "port série" sur forum C), mais je tombe sur des codes Windows que je n'arrive pas à décrypter. Ci joint mon code actuel :

    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
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    #ifdef HAVE_CONFIG_H
    #include <config.h>
    #endif
     
    #include <stdlib.h>
    #include <stdio.h>
    #include <string.h>
    #include <unistd.h>
    #include <fcntl.h>
    #include <errno.h>
    #include <termios.h>
     
    int srl_handle;
    struct termios options;
     
    void mode_raw(int activer)
    {
    	struct termios raw;
    	static int raw_actif=0;
    	static struct termios cooked;
     
    	if (raw_actif == activer) return;
    	if (activer)
    	{
    		tcgetattr(STDIN_FILENO, &cooked);
    		raw = cooked;
    		cfmakeraw(&raw);
    		tcsetattr(STDIN_FILENO, TCSANOW, &raw);
    		printf("Raw mode activated\r\n");
    	}
    	else
    	{
    		tcsetattr(STDIN_FILENO, TCSANOW, &cooked);
    		printf("Raw mode deactivated\r\n");
    	}
     
    	raw_actif = activer;
    }
     
    int main(int argc, char *argv[])
    {
    	char c=0;
    	char s[2]="\0";
    	char* buf=0;
     
    	mode_raw(1);
    	/*Open serial port*/
    	srl_handle=open("/dev/ttyS0", O_RDWR | O_NOCTTY | O_NDELAY);
    	if(srl_handle<0)
    	{
    		perror("serial port open");
    		exit(-1);
    	}
    	fcntl(srl_handle, F_SETFL, FNDELAY);
    	tcgetattr(srl_handle, &options);
     
    	/*Set baud rate*/
    	cfsetospeed(&options, B19200);
    	cfsetispeed(&options, B19200);
    	options.c_cflag |= (CLOCAL | CREAD);
    	/*Set no parity*/
    	options.c_cflag |= PARENB;
    	options.c_cflag &= ~CSTOPB;
    	options.c_cflag &= ~CSIZE;
    	options.c_cflag &= ~PARODD;
    	options.c_cflag |= CS7;
    	options.c_cflag &= ~CRTSCTS;
    	options.c_iflag &= ~(IXON|IXOFF|IXANY);
    	/*Set raw output*/
    	tcsetattr(srl_handle, TCSANOW, &options);
     
    	/*Write Data*/
    	while ((c=fgetc(stdin))!='x')
    	{
    		printf("%c\t",c);
    		s[0] = c;
    		s[1] = '\0';
    		printf("%s\r\n",s);
    		if (write(srl_handle, s, 1) < 0)
    			fputs("write() failed!\r\n", stderr);
    	}
    	mode_raw(0);
    //	fcntl(srl_handle, F_SETFL, 0);
    //	while (strstr(buf,"OK")==NULL)
    //	{
    		if (read(srl_handle, buf, 10) < 0)
    			fputs("read() failed!\r\n", stderr);
    		printf("%s\n",buf);
    //	}
     
    	close(srl_handle);	
    //	mode_raw(0);
    	return EXIT_SUCCESS;
     
    }
    C'est cette partie qui ne semble pas fonctionner, en tout cas, je n'ai pas l'impression de recevoir quoi que ce soit :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
     
    //	fcntl(srl_handle, F_SETFL, 0);
    //	while (strstr(buf,"OK")==NULL)
    //	{
    		if (read(srl_handle, buf, 10) < 0)
    			fputs("read() failed!\r\n", stderr);
    		printf("%s\n",buf);
    //	}
    car lorsque je décommente la ligne
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    //	fcntl(srl_handle, F_SETFL, 0);
    je suis sensé attendre la lecture d'un caractère sur mon port série, mais rien ne vient et mon programme reste bloqué.

    Please help

    Merci

    DBoJ

  2. #2
    Expert éminent sénior

    Profil pro
    Inscrit en
    Janvier 2007
    Messages
    10 603
    Détails du profil
    Informations personnelles :
    Âge : 66
    Localisation : France

    Informations forums :
    Inscription : Janvier 2007
    Messages : 10 603
    Points : 17 913
    Points
    17 913
    Billets dans le blog
    2
    Par défaut
    ben buf n'est pas dimensionné

    Essaye de mettre

    char buf[500] ;
    "Un homme sage ne croit que la moitié de ce qu’il lit. Plus sage encore, il sait laquelle".

    Consultant indépendant.
    Architecture systèmes complexes. Programmation grosses applications critiques. Ergonomie.
    C, Fortran, XWindow/Motif, Java

    Je ne réponds pas aux MP techniques

  3. #3
    Nouveau membre du Club
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    39
    Détails du profil
    Informations personnelles :
    Âge : 44
    Localisation : France

    Informations forums :
    Inscription : Octobre 2006
    Messages : 39
    Points : 32
    Points
    32
    Par défaut
    ben buf n'est pas dimensionné
    Essaye de mettre
    char buf[500] ;
    Ca doit pas être ça, le comportement ne change pas. En plus, je pensais qu'avec un char* ça suffisait, car la fonction read() demande un void* en paramètre.

    Autre question sur le même projet, actuellement comme on peut le lire, je fais une boucle qui va écrire sur la liaison série puis qui va lire, et ça en permanence. J'aimerais savoir comment me placer dans une boucle d'attente et ne lire le port série que quand je reçois des données. J'ai vu qu'il existait un signal SIGIO, qu'on pouvait triater avec un handler, mais je ne sais pas comment faire et ça me semble très nébuleux. Si quelqu'un avait un exemple concret pour m'expliquer, ça m'intéresserait.

    DBoJ

  4. #4
    Expert éminent sénior

    Profil pro
    Inscrit en
    Janvier 2007
    Messages
    10 603
    Détails du profil
    Informations personnelles :
    Âge : 66
    Localisation : France

    Informations forums :
    Inscription : Janvier 2007
    Messages : 10 603
    Points : 17 913
    Points
    17 913
    Billets dans le blog
    2
    Par défaut
    Citation Envoyé par DangerousBowlOfJelly Voir le message
    Ca doit pas être ça, le comportement ne change pas. En plus, je pensais qu'avec un char* ça suffisait, car la fonction read() demande un void* en paramètre.
    revoie les bases du C.

    Quand une fonction attend un pointeur (vide ou non), ca ne veut pas dire que l'adresse et la taille doivent etre indefinies.....

    read attend un buffer, et bien entendu on passe l'adresse du buffer..

    Mais ce pourrait etre une structure, un tableau de stuctures, etc.. D'ou le void...

    Quant a l'attente, un sleep devrait suffire.. L'arrivee d'un signal sur le port doit reveiller le sleep

    donc

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    while ( 1 )
    {
      sleep (9999);
     
      read.....
       ....
    }
    "Un homme sage ne croit que la moitié de ce qu’il lit. Plus sage encore, il sait laquelle".

    Consultant indépendant.
    Architecture systèmes complexes. Programmation grosses applications critiques. Ergonomie.
    C, Fortran, XWindow/Motif, Java

    Je ne réponds pas aux MP techniques

  5. #5
    Expert éminent sénior
    Avatar de Emmanuel Delahaye
    Profil pro
    Retraité
    Inscrit en
    Décembre 2003
    Messages
    14 512
    Détails du profil
    Informations personnelles :
    Âge : 67
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Retraité

    Informations forums :
    Inscription : Décembre 2003
    Messages : 14 512
    Points : 20 985
    Points
    20 985
    Par défaut
    Citation Envoyé par DangerousBowlOfJelly Voir le message
    Ca doit pas être ça, le comportement ne change pas.
    Ce n'est peut être pas le seul problème, mais il doit être corrigé.
    En plus, je pensais qu'avec un char* ça suffisait, car la fonction read() demande un void* en paramètre.
    Et ? A ton avis, le char * en question pointe sur quoi ?
    Pas de Wi-Fi à la maison : CPL

  6. #6
    Nouveau membre du Club
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    39
    Détails du profil
    Informations personnelles :
    Âge : 44
    Localisation : France

    Informations forums :
    Inscription : Octobre 2006
    Messages : 39
    Points : 32
    Points
    32
    Par défaut
    Bonjour.

    Je reprends mon fil pour ne pas en ajouter un autre, car c'est toujours le même port que je cherche à faire fonctionner.

    La communication fonctionne très bien, je reçois tout ce que je veux et j'envoie correctement.

    Maintenant, j'aimerais savoir comment on peut contrôler l'état des lignes RTS/CTS ou DTR/DSR pour pouvoir détecter une coupure de ligne lors d'un transfert de fichier. Actuellement, j'attends indéfiniment un caractère 0x03 (ETX) et je regarde si ce que j'ai sauvegardé correspond à la taille spécifiée.

    Merci de vos réponses.

    DBoJ

  7. #7
    Expert éminent sénior
    Avatar de Emmanuel Delahaye
    Profil pro
    Retraité
    Inscrit en
    Décembre 2003
    Messages
    14 512
    Détails du profil
    Informations personnelles :
    Âge : 67
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Retraité

    Informations forums :
    Inscription : Décembre 2003
    Messages : 14 512
    Points : 20 985
    Points
    20 985
    Par défaut
    Citation Envoyé par DangerousBowlOfJelly Voir le message
    Maintenant, j'aimerais savoir comment on peut contrôler l'état des lignes RTS/CTS ou DTR/DSR pour pouvoir détecter une coupure de ligne lors d'un transfert de fichier. Actuellement, j'attends indéfiniment un caractère 0x03 (ETX) et je regarde si ce que j'ai sauvegardé correspond à la taille spécifiée.
    Probablement avec une des fonctions de termios ou des sous-fonctions de ioctl(). Mais laquelle, aucune idée. C'est expliqué dans la doc de programmation de Linux...

    man termios etc.
    Pas de Wi-Fi à la maison : CPL

Discussions similaires

  1. Problème de ports série sous linux
    Par Imaging dans le forum Matériel
    Réponses: 0
    Dernier message: 15/05/2009, 19h24
  2. [Série] Accès au port série sous linux
    Par ghost dans le forum Entrée/Sortie
    Réponses: 10
    Dernier message: 10/10/2007, 10h43
  3. Pb lecture sur port série sous XP SP2
    Par ViveLesQuads dans le forum Delphi
    Réponses: 7
    Dernier message: 09/06/2006, 17h12
  4. Port série sous Linux
    Par naoned dans le forum Bibliothèques tierces
    Réponses: 2
    Dernier message: 21/11/2005, 16h45
  5. [C#] lecture sur port série
    Par Iokanaan dans le forum Windows Forms
    Réponses: 3
    Dernier message: 03/05/2005, 13h50

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