Communication RS232 (programmation)

J'en ai vraiment marre... je pensais avoir terminer mon code avec tes précieux conseils et voilà que je viens de m'apercevoir que la PWM des pins 10,9 et la PWM des pins 6,5 ne sont pas cadencés de manière identique... :confused:

Quand j'observe au scope j'ai une fréquence de environ 490Hz pour les pins 10 et 9 et j'ai le double pour les 6 et 5.

Alors il me vient deux quetions :

1/ Est-ce que ça sera la même chose sur la Nano (car je développe mon code sur une Uno)
2/ Est-ce que c'est possible de la changer facilement ? J'ai regardé un peu... :fearful:

rhum187:
1/ Est-ce que ça sera la même chose sur la Nano (car je développe mon code sur une Uno)

Oui c'est une différence logiciel liée au "core arduino", pas à la carte.

rhum187:
2/ Est-ce que c'est possible de la changer facilement ? J'ai regardé un peu... :fearful:

Oui et non

Regarde ici :
http://arduino.cc/playground/Code/PwmFrequency

Ok,

ce qui veut dire que je ne peux ni faire tomber les pins 5 et 6 à 490Hz ni monter les pins 9 et 10 à 970Hz car il manque le case 128

rhum187:
ce qui veut dire que je ne peux ni faire tomber les pins 5 et 6 à 490Hz ni monter les pins 9 et 10 à 970Hz car il manque le case 128

C'est vraiment un point critique que tes PWM est une fréquence identique ?

Ben je me posais la question justement...

Je pense que le composant auquel je les envoie s'en moque un peu...

EDIT : C'est la ds de mon pont en H : http://www.ti.com/lit/ds/symlink/drv8432.pdf

Yep!

Si çà peut aider, tu peux piloter autant de servos que tu veux avec un seul PWM et sur n'importe quelle pinoche.

Je viens de plus ou moins faire le tour ce week-end de mes servos :wink:

@+

Zoroastre.

Merci zoroastre,

J'ai regardé ton topic avec beaucoup d'attention et ça pourra éventuellement m'aider :wink:

Je viens de rentrer dans la 4ème dimensions... ^^

Non, pas encore, malheureusement, il y a un problème que je n'arrive pas à résoudre.

Je fais communiquer deux Arduino. Avec le premier j'envoie ça :

  Serial.print('M'); // Permet l'identification des commandes moteurs par le programme robot
  Serial.write(ZONE); // Permet le débogage des conditions à l'aide du VI
  Serial.write((int)PWM1);
  Serial.write((int)PWM2);
  Serial.write((int)PWM3);
  Serial.write((int)PWM4);

et avec le deuxième je veux lire la valeur des PWM donc j'ai écrit (avec l'aide de Skywodd ;)) :

byte PWM_MOT[4]; // Tableau de 6 octets
int n_octet = 4;
int pin_PWM_A = 10;
int pin_PWM_B = 9;
int pin_PWM_C = 6;
int pin_PWM_D = 5;

void setup()
{
  Serial.begin(9600); // Paramétrage de la vitesse de la liaison série
}

/*********************** DEBUT DE LA BOUCLE PRINCIPALE ********************************************/
void loop()
{ 
  if(Serial.available()>0)
  {
    if(Serial.read()!='M')
    {
     delay(50);
    }
    else
    {
      for(int n=1; n<=n_octet; n++)
      {
        while(Serial.available() < 1);
        PWM_MOT[n]=Serial.read();
      }
    }
    //analogWrite(pin_PWM_A, PWM_MOT[1]);
    //analogWrite(pin_PWM_B, PWM_MOT[2]);
    //analogWrite(pin_PWM_C, PWM_MOT[3]);
    //analogWrite(pin_PWM_D, PWM_MOT[4]);
    
    //Serial.print(PWM_MOT[0]);
    Serial.print(PWM_MOT[1]);
    Serial.print(PWM_MOT[2]);
    Serial.print(PWM_MOT[3]);
    Serial.print(PWM_MOT[4]);
    //Serial.print(PWM_MOT[5]);
    
    Serial.println();
    
  }
  else
  {
    analogWrite(pin_PWM_A, 0);
    analogWrite(pin_PWM_B, 0);
    analogWrite(pin_PWM_C, 0);
    analogWrite(pin_PWM_D, 0);
  }
  
  Serial.flush();
  delay(50);
}

Et je vous joins les deux screen XCTU qui m'ont permis de constater que il y avait +1 sur la PWM4 alors que je n'ai rien fait pour... enfin je pense
COM11 : Arduino qui envoit les infos
COM13 : Arduino qui reçoit et qui les (ré)affichent

rhum187:
Ben je me posais la question justement...

Je pense que le composant auquel je les envoie s'en moque un peu...

Ça n'as aucune réelle importance, dans le pire des cas cela ne joue que sur le sifflement / grésillement résiduelle du moteur en sortie (si tu utilise un moteur).

Oui je pense aussi.

Par contre ma différence de 1, même si ce n'est que 1 me perturbe un peu plus...

rhum187:
Par contre ma différence de 1, même si ce n'est que 1 me perturbe un peu plus...

Rooo! Je sort le fusil et les cartouches si tu ne vois pas tes erreurs :wink:

Erreur 1 :
Prototype :

Serial.write(uint8_t);

Ton code :

Serial.write((int)PWM4);

Note : uint8_t = 1 octet non signé (0 ~ 255), int = 2 octets signé (-32767 ~ 32767)

--

Erreur 2 :
Déclaration :

byte PWM_MOT[4]; // Tableau de 6 octets

(hummm 6 tu est sûr ;))

Ton code ensuite :

//Serial.print(PWM_MOT[0]);
    Serial.print(PWM_MOT[1]); // <-
    Serial.print(PWM_MOT[2]);
    Serial.print(PWM_MOT[3]);
    Serial.print(PWM_MOT[4]); // <- 
    //Serial.print(PWM_MOT[5]);

Note : en C/C++ les tableaux commencent à 0 et se terminent à taille_max - 1, exemple int[2] -> int[0], int[1]

--

Erreur 3 :
Déclaration :

byte PWM_MOT[4];
int n_octet = 4;

Ton code :

for(int n=1; n<=n_octet; n++) // -> n = 1, n = 2, n = 3, n = 4
{
  while(Serial.available() < 1);
  PWM_MOT[n]=Serial.read(); // <- 
}

Voir note plus haut.

Erreur 4 :

Tu envois 6 octets mais tu en lit seulement 4 (+ 1 pour le if) :wink:

--

Correction :

Envoi :

Serial.write((uint8_t)'M'); // Permet l'identification des commandes moteurs par le programme robot
Serial.write((uint8_t)ZONE); // Permet le débogage des conditions à l'aide du VI
Serial.write((uint8_t)PWM1);
Serial.write((uint8_t)PWM2);
Serial.write((uint8_t)PWM3);
Serial.write((uint8_t)PWM4);

Réception :

const byte NB_OCTETS = 5;
const byte PIN_PWM_A = 10;
const byte PIN_PWM_B = 9;
const byte PIN_PWM_C = 6;
const byte PIN_PWM_D = 5;

byte PWM_MOT[NB_OCTETS];

void setup() {
  Serial.begin(9600); // Paramétrage de la vitesse de la liaison série
}

/*********************** DEBUT DE LA BOUCLE PRINCIPALE ********************************************/
void loop() { 
  
  if(Serial.available() > 0) {
    
    if(Serial.read() != 'M') {
      delay(50);
    } 
    else {

      for(int n = 0; n < NB_OCTETS; ++n) {
        while(Serial.available() < 1);
        PWM_MOT[n] = Serial.read();
      }

      //analogWrite(PIN_PWM_A, PWM_MOT[1]);
      //analogWrite(PIN_PWM_B, PWM_MOT[2]);
      //analogWrite(PIN_PWM_C, PWM_MOT[3]);
      //analogWrite(PIN_PWM_D, PWM_MOT[4]);

      Serial.print(PWM_MOT[0]);
      Serial.print(PWM_MOT[1]);
      Serial.print(PWM_MOT[2]);
      Serial.print(PWM_MOT[3]);
      Serial.print(PWM_MOT[4]);
      Serial.println();
    } 
  } 
  else {
    
    analogWrite(PIN_PWM_A, 0);
    analogWrite(PIN_PWM_B, 0);
    analogWrite(PIN_PWM_C, 0);
    analogWrite(PIN_PWM_D, 0);
  }

  // Serial.flush(); // Pas de flush() quand on ne sait pas ce que ça signifie ;)
  delay(50);
}

Pas de flush() quand on ne sait pas ce que ça signifie :wink:

Heu, pour moi je vidais le buffer de lecture à chaque fois que j'ai terminé mais je me suis rendu compte que je mettre ou pas n'avais aucun effet...

Ca a du te faire pas mal de boulot alors merci pour ton temps je regarde ça et je te fais un retour avant 19h (si mon chef me sort pas une réunion surprise du chapeau...)

rhum187:
Heu, pour moi je vidais le buffer de lecture à chaque fois que j'ai terminé mais je me suis rendu compte que je mettre ou pas n'avais aucun effet...

Justement tu as tout faux :wink:
Depuis arduino 1.0 Serial.flush() bloque le programme jusqu'à ce que toute les données en Tx soit émise.

Pour vider le buffer entrant il faut faire une boucle de ce genre :

while(Serial.available() > 0) Serial.read();

rhum187:
Ca a du te faire pas mal de boulot alors merci pour ton temps je regarde ça et je te fais un retour avant 19h (si mon chef me sort pas une réunion surprise du chapeau...)

Bof juste 5 minutes entre deux bricolages 8)

Depuis arduino 1.0

Je bosse encore sur la 0023

rhum187:

Depuis arduino 1.0

Je bosse encore sur la 0023

Ce qui ne change rien au fait que flush() n'as rien à faire dans ton prog :wink:

Ok,

bon ben écoute sans surprise ça fonctionne beaucoup mieux avec les uint8_t

(hummm 6 tu est sûr )

Non bien sur mais je n'avais pas maj mon commentaire... ça reste une erreur me dirais mes profs... XD

(+ 1 pour le if)

tu voulais dire le :

while(Serial.available() < 1);

que tu m'as fait rajouté... :wink:

while(Serial.available() > 0) Serial.read();

Ha ok, le fait de faire un read vide le buffer si je comprends bien !

const byte NB_OCTETS = 5;
const byte PIN_PWM_A = 10;
const byte PIN_PWM_B = 9;
const byte PIN_PWM_C = 6;
const byte PIN_PWM_D = 5;

Et une petite question au passage : pourquoi avoir choisi de déclarer ça en const byte ?

Ma supposition : const veut dire que ça ne bouge pas (c'est bien le propre d'une constante)
byte car un octet suffit pour coder ces valeurs

J'ai bon ? j'ai bon ? :slight_smile:

rhum187:
Ma supposition : const veut dire que ça ne bouge pas (c'est bien le propre d'une constante)
byte car un octet suffit pour coder ces valeurs

J'ai bon ? j'ai bon ? :slight_smile:

C'est ça.

Const = constante = qui ne change pas (si on tente de changer la valeur d'un const = erreur de compilation)
byte = 1 octet = économie de ram vu que l'on stocke juste un nombre avec pour valeur max 255.

Parfait ben plus qu'à faire les deux modifs pour juste afficher les 4 PWM et ça devrait être bon avec qqch du style :

const byte NB_OCTETS = 4;
.
.
.
byte PWM_MOT[NB_OCTETS-1];
.
.
.
for(int n = 2; n < NB_OCTETS+2; ++n) 
      {
        while(Serial.available() < 1);
        PWM_MOT[n] = Serial.read();
      }

Alors voyons, chevrotine pour gros gibier ou cartouches, hummm ... :grin:

for(int n = 2; n < NB_OCTETS+2; ++n) // n = 2, n = 3, n = 4, n = 5
{
  while(Serial.available() < 1);
  PWM_MOT[n] = Serial.read();
}

Les tableaux en C/C++ commencent à zéro !
(Souligné, gras, surligné, je peut pas faire plus lisible ;))

Correction :

Serial.read();  // Laisse de côté l'octet "ZONE" (est-ce bien utile de l'envoyer du coup ? Pour moi non ;))
for(byte n = 0; n < NB_OCTETS - 1; ++n)  {
  while(Serial.available() < 1);
  PWM_MOT[n] = Serial.read();
}