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

Vous êtes nouveau sur Developpez.com ? Créez votre compte ou connectez-vous afin de pouvoir participer !

Vous devez avoir un compte Developpez.com et être connecté pour pouvoir participer aux discussions.

Vous n'avez pas encore de compte Developpez.com ? Créez-en un en quelques instants, c'est entièrement gratuit !

Si vous disposez déjà d'un compte et qu'il est bien activé, connectez-vous à l'aide du formulaire ci-dessous.

Identifiez-vous
Identifiant
Mot de passe
Mot de passe oublié ?
Créer un compte

L'inscription est gratuite et ne vous prendra que quelques instants !

Je m'inscris !

Apprendre à piloter votre Arduino depuis le Moniteur Série avec SerialCommands
Un billet blog de f-leb

Le , par f-leb

0PARTAGES

Vous voudriez piloter votre Arduino en tapant des commandes depuis le Moniteur Série ? Pour cela, au lieu de réinventer la roue, utilisez plutôt une bibliothèque comme SerialCommands, simple de fonctionnement et avec une faible empreinte mémoire. Il est en effet assez fréquent d'avoir besoin d'ajuster un ou plusieurs paramètres en fonction d'événements sur votre système embarqué en cours de fonctionnement. Plutôt que de modifier la valeur d'un paramètre codée en dur dans le programme (un seuil, un coefficient P, I ou D de votre régulateur, la vitesse d'un moteur, etc.), compiler et téléverser à nouveau le programme, et recommencer la procédure si nécessaire, modifiez vos paramètres en direct et de façon interactive depuis le Moniteur Série.

Comme démonstration, je prendrais le petit programme de clignotement de LED RGB (pour faire original) ci-dessous. Ma LED RGB (à cathode commune) se mettra simplement à clignoter (fréquence 1 Hz), et quand elle sera allumée, elle sera de couleur magenta (parce que c'est plus joli !)

demo-serialCommands-rgb.ino
Code arduino : Sélectionner tout
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
#include "ledRGB.h" 
  
// LedRGB ne gère que les led RGB à cathode commune !! 
LedRGB myLed(9, 11, 10); // rouge=pin 9, vert=pin 11, et bleu=pin 10 supportent le PWM sur Arduino Uno 
  
byte colorOn = Color::magenta; // équivalent à colorOn = Color::red | Color::blue 
int brightness = 100 ; // luminosité en % 
  
byte ledState = Color::off;   // état de la Led, initialement éteinte 
  
unsigned long previousMillis = millis(); 
const long interval = 500;  // intervalle = 500ms 
  
void setup() {   
  
} 
  
void loop() { 
  unsigned long currentMillis = millis(); 
  
  if ((currentMillis - previousMillis) > interval) { // si intervalle de temps écoulé 
    ledState = (ledState == Color::off) ? colorOn : Color::off; // bascule led On/Off 
    myLed.digitalWrite_RGB(ledState, brightness); 
    previousMillis = currentMillis; 
  } 
}


ledRGB.h




Code arduino : Sélectionner tout
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
#ifndef H_LED_RGB 
#define H_LED_RGB 
  #include <Arduino.h> 
  
  enum Color {  red   = 1 << 2, 
                green = 1 << 1, 
                blue  = 1 << 0, 
                yellow = red | green, 
                cyan = green | blue, 
                magenta = red | blue, 
                white = red | green | blue, 
                off = 0 
             }; 
  
  class LedRGB{ 
    public:        
      LedRGB(byte pRed, byte pGreen, byte pBlue); 
      void digitalWrite_RGB(byte ledColor); 
      void digitalWrite_RGB(byte ledColor, int brightness); 
  
    private: 
      byte _pinRed, _pinGreen, _pinBlue; 
    }; 
#endif




ledRGB.cpp




Code arduino : Sélectionner tout
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
#include "ledRGB.h" 
  
 LedRGB::LedRGB(byte pRed, byte pGreen, byte pBlue){ 
        _pinRed = pRed; 
        _pinGreen = pGreen; 
        _pinBlue = pBlue; 
        pinMode(_pinRed, OUTPUT); 
        pinMode(_pinGreen, OUTPUT); 
        pinMode(_pinBlue, OUTPUT);        
        } 
  
 void LedRGB::digitalWrite_RGB(byte ledColor) { 
        digitalWrite(_pinRed  , (ledColor & Color::red) ? HIGH : LOW); 
        digitalWrite(_pinGreen, (ledColor & Color::green) ? HIGH : LOW); 
        digitalWrite(_pinBlue , (ledColor & Color::blue) ? HIGH : LOW); 
      } 
  
void LedRGB::digitalWrite_RGB(byte ledColor, int brightness) { 
        analogWrite(_pinRed  , 255*((ledColor & Color::red) ? brightness : 0) / 100); 
        analogWrite(_pinGreen, 255*((ledColor & Color::green) ? brightness : 0) / 100); 
        analogWrite(_pinBlue , 255*((ledColor & Color::blue) ? brightness : 0) / 100);   
      }




Supposons que je veuille régler la couleur de ma LED qui clignote, ainsi que sa luminosité. Ces deux paramètres sont dans des variables globales du fichier principal avec des valeurs initiales non satisfaisantes, et qu'il faudra ajuster en cours de fonctionnement :
Code arduino : Sélectionner tout
1
2
byte colorOn = Color::magenta; // équivalent à colorOn = Color::red | Color::blue 
int brightness = 100 ; // luminosité en %

Par exemple avec la commande rgb suivante à saisir dans le Moniteur Série, suivie de 3 ou 4 paramètres :
rgb 0 0 0 ==> LED éteinte
rgb 1 0 0 ==> rouge
rgb 0 1 0 ==> vert
rgb 0 0 1 ==> bleu
rgb 1 1 0 ==> rouge + vert = jaune
rgb 0 1 1 ==> vert + bleu = cyan
rgb 1 0 1 ==> rouge + bleu = magenta
rgb 1 1 1 ==> rouge + vert + bleu = magenta
rgb 1 1 0 50 ==> rouge + vert = jaune, brillance = 50%

Et si vous avez oublié la syntaxe des commandes ou comment fonctionne la synthèse additive des couleurs primaires, la commande help vous fera un rappel.

Tant qu'à faire, l'implémentation de ces commandes série ne doit pas remettre en cause le programme principal, en le modifiant le moins possible.
Le programme principal modifié devient (modifications en rouge) :

demo-serialCommands-rgb.ino
Code arduino : Sélectionner tout
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
#include "ledRGB.h" 
#include "ledSerialCommands.h" 
 
// LedRGB ne gère que les led RGB à cathode commune !! 
LedRGB myLed(9, 11, 10); // rouge=pin 9, vert=pin 11, et bleu=pin 10 supportent le PWM sur Arduino Uno 
 
byte colorOn = Color::magenta; // équivalent à colorOn = Color::red | Color::blue 
int brightness = 100 ; // luminosité en % 
 
byte ledState = Color::off;   // état de la Led, initialement éteinte 
 
unsigned long previousMillis = millis(); 
const long interval = 500;  // intervalle = 500ms 
 
void setup() {   
  Serial.begin(115200); 
  init_serial_commands(); 
} 
 
void loop() { 
 
  update_serial_commands(); 
 
  unsigned long currentMillis = millis(); 
 
  if ((currentMillis - previousMillis) > interval) { // si intervalle de temps écoulé 
    ledState = (ledState == Color::off) ? colorOn : Color::off; // bascule led On/Off 
    myLed.digitalWrite_RGB(ledState, brightness); 
    previousMillis = currentMillis; 
  } 
}

Fonctionnement de SerialCommands

Pour déclarer un gestionnaire général pour les commandes série :
Code arduino : Sélectionner tout
1
2
char serial_command_buffer_[17]; 
SerialCommands serial_commands_(&Serial, serial_command_buffer_, sizeof(serial_command_buffer_), "\r\n", " ");
  • Pour la communication série, l'adresse de l'objet Serial est passée en paramètre.
  • Terminaison des commandes avec \r\n (NewLine et Carriage Return).
  • Séparateur entre les paramètres, caractère <espace> : " ".


Pour déclarer les commandes :
Code arduino : Sélectionner tout
1
2
3
4
SerialCommand cmd_rgb_("rgb", cmd_rgb); 
SerialCommand cmd_brightness_down_("-", cmd_brightness_down, true); // commande "one key" 
SerialCommand cmd_brightness_up_("+", cmd_brightness_up, true);     // commande "one key" 
SerialCommand cmd_help_("help", cmd_help);
En paramètre de chaque commande déclarée, la commande telle qu'elle doit être saisie dans le Moniteur Série et la fonction de rappel associée.
Lorsque le troisième paramètre est à true, on déclare une commande sur une touche (one key). Ici les commandes + et - permettent d'augmenter ou diminuer la luminosité.

Pour ajouter les commandes au gestionnaire, appel depuis le setup() :
Code arduino : Sélectionner tout
1
2
3
4
5
serial_commands_.AddCommand(&cmd_rgb_); 
serial_commands_.AddCommand(&cmd_brightness_down_); 
serial_commands_.AddCommand(&cmd_brightness_up_); 
serial_commands_.AddCommand(&cmd_help_); 
serial_commands_.SetDefaultHandler(cmd_unrecognized);

Pour appeler le gestionnaire, appel depuis la boucle loop() :
Code arduino : Sélectionner tout
serial_commands_.ReadSerial();

Dans les fonctions de rappel de chaque commande, on récupère le ou les paramètres qui suivent en appelant à chaque fois la méthode Next() :
Code arduino : Sélectionner tout
char* parameter_str = sender->Next();  // paramètre suivant

Le projet final :






demo-serialCommands-rgb.ino
Code arduino : Sélectionner tout
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
#include "ledRGB.h" 
#include "ledSerialCommands.h" 
  
// ledRGB ne gère que les led RGB à cathode commune !! 
LedRGB myLed(9, 11, 10); // rouge=pin 9, vert=pin 11, et bleu=pin 10 supportent le PWM sur Arduino Uno 
  
byte colorOn = Color::magenta; // équivalent à colorOn = Color::red | Color::blue 
int brightness = 100 ; // luminosité en % 
  
byte ledState = Color::off;   // état de la Led, initialement éteinte 
  
unsigned long previousMillis = millis(); 
const long interval = 500;  // intervalle = 500ms 
  
void setup() {   
  Serial.begin(115200); 
  init_serial_commands(); 
} 
  
void loop() { 
  update_serial_commands(); 
  
  unsigned long currentMillis = millis(); 
  
  if ((currentMillis - previousMillis) > interval) { // si intervalle de temps écoulé 
    ledState = (ledState == Color::off) ? colorOn : Color::off; // bascule led On/Off 
    myLed.digitalWrite_RGB(ledState, brightness); 
    previousMillis = currentMillis; 
  } 
}

ledRGB.h
Code arduino : Sélectionner tout
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
#ifndef H_LED_RGB 
#define H_LED_RGB 
  #include <Arduino.h> 
  
  enum Color {  red   = 1 << 2, 
                green = 1 << 1, 
                blue  = 1 << 0, 
                yellow = red | green, 
                cyan = green | blue, 
                magenta = red | blue, 
                white = red | green | blue, 
                off = 0 
             }; 
  
  class LedRGB{ 
    public:        
      LedRGB(byte pRed, byte pGreen, byte pBlue); 
      void digitalWrite_RGB(byte ledColor); 
      void digitalWrite_RGB(byte ledColor, int brightness); 
  
    private: 
      byte _pinRed, _pinGreen, _pinBlue; 
    }; 
#endif

ledRGB.cpp
Code arduino : Sélectionner tout
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
#include "ledRGB.h" 
  
 LedRGB::LedRGB(byte pRed, byte pGreen, byte pBlue){ 
        _pinRed = pRed; 
        _pinGreen = pGreen; 
        _pinBlue = pBlue; 
        pinMode(_pinRed, OUTPUT); 
        pinMode(_pinGreen, OUTPUT); 
        pinMode(_pinBlue, OUTPUT);        
        } 
  
 void LedRGB::digitalWrite_RGB(byte ledColor) { 
        digitalWrite(_pinRed  , (ledColor & Color::red) ? HIGH : LOW); 
        digitalWrite(_pinGreen, (ledColor & Color::green) ? HIGH : LOW); 
        digitalWrite(_pinBlue , (ledColor & Color::blue) ? HIGH : LOW); 
      } 
  
void LedRGB::digitalWrite_RGB(byte ledColor, int brightness) { 
        analogWrite(_pinRed  , 255*((ledColor & Color::red) ? brightness : 0) / 100); 
        analogWrite(_pinGreen, 255*((ledColor & Color::green) ? brightness : 0) / 100); 
        analogWrite(_pinBlue , 255*((ledColor & Color::blue) ? brightness : 0) / 100);   
      }

ledSerialCommands.h
Code arduino : Sélectionner tout
1
2
3
4
5
6
7
8
9
10
11
12
13
#ifndef H_LED_SERIAL_COMMANDS 
#define H_LED_SERIAL_COMMANDS 
  #include <SerialCommands.h> //https://www.arduino.cc/reference/en/libraries/serialcommands/ 
  
  void init_serial_commands(void); 
  void update_serial_commands(void); 
  
  void cmd_rgb(SerialCommands* sender); 
  void cmd_brightness_down(SerialCommands* sender); 
  void cmd_brightness_up(SerialCommands* sender); 
  void cmd_help(SerialCommands* sender); 
  void cmd_unrecognized(SerialCommands* sender, const char* cmd);  // la fonction appelée si la commande saisie est inconnue 
#endif

ledSerialCommands.cpp
Code arduino : Sélectionner tout
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
96
97
#include "ledSerialCommands.h" 
#include "ledRGB.h" 
  
extern byte colorOn; 
extern int brightness; 
  
char serial_command_buffer_[17]; 
/* Gestionnaire général pour les commandes série 
   Pour la communication série, l'adresse de l'objet Serial est passée en paramètre 
   Terminaison des commandes \r\n, NewLine et Carriage Return 
   Séparateur entre les paramètres, caractère espace " "                                                 */ 
SerialCommands serial_commands_(&Serial, serial_command_buffer_, sizeof(serial_command_buffer_), "\r\n", " "); 
  
SerialCommand cmd_rgb_("rgb", cmd_rgb); 
SerialCommand cmd_brightness_down_("-", cmd_brightness_down, true); // commande "one key" 
SerialCommand cmd_brightness_up_("+", cmd_brightness_up, true);     // commande "one key" 
SerialCommand cmd_help_("help", cmd_help); 
  
// ----------------------------------------------------------------------------- 
void cmd_rgb(SerialCommands* sender) 
{ 
  int tab[3], i = 0; 
  bool error = false; 
  while (i < 3 && !error) { 
    char* parameter_str = sender->Next();  // paramètre suivant 
    if (strcmp(parameter_str, "0") ==  0 || strcmp(parameter_str, "1") == 0) { 
      tab[i] = atoi(parameter_str); 
      i++; 
    } 
    else { 
      error = true; 
    } 
  } 
  
  if (!error) { 
    colorOn = ((tab[0] == 1) ? Color::red   : 0) | 
              ((tab[1] == 1) ? Color::green : 0) | 
              ((tab[2] == 1) ? Color::blue  : 0); 
  
    char* parameter_str = sender->Next(); // si paramètre supplémentaire, c'est le paramètre de brillance 
    if (parameter_str != NULL) { 
      brightness = atoi(parameter_str); 
    } 
  } 
  else {  // Si erreur 
    sender->GetSerial()->println(F("Erreur de syntaxe")); 
    sender->GetSerial()->println(F("Voir l'aide en tapant la commande [help]")); 
  } 
} 
  
void cmd_brightness_down(SerialCommands* sender) 
{ 
  brightness -= 5; 
  if (brightness < 0) brightness = 0; 
  
  sender->GetSerial()->print(F("brightness = ")); 
  sender->GetSerial()->println(brightness); 
} 
  
void cmd_brightness_up(SerialCommands* sender) 
{ 
  brightness += 5; 
  if (brightness > 100) brightness = 100; 
  
  sender->GetSerial()->print(F("brightness = ")); 
  sender->GetSerial()->println(brightness); 
} 
  
void cmd_help(SerialCommands* sender) 
{ 
  sender->GetSerial()->println(F("Exemples de commandes :")); 
  sender->GetSerial()->println(F("[rgb 1 0 1]  --> rouge + bleu = magenta")); 
  sender->GetSerial()->println(F("[rgb 1 1 0]  --> rouge + vert = jaune")); 
  sender->GetSerial()->println(F("[rgb 1 1 0 50]  --> rouge + vert = jaune, brillance = 50%")); 
} 
  
void cmd_unrecognized(SerialCommands* sender, const char* cmd) 
{ 
  sender->GetSerial()->print(F("Commande inconnue [")); 
  sender->GetSerial()->print(cmd); 
  sender->GetSerial()->println(F("]")); 
} 
  
void init_serial_commands(void) 
{ 
  /* Ajout des commandes au gestionnaire principal */ 
  serial_commands_.AddCommand(&cmd_rgb_); 
  serial_commands_.AddCommand(&cmd_brightness_down_); 
  serial_commands_.AddCommand(&cmd_brightness_up_); 
  serial_commands_.AddCommand(&cmd_help_); 
  serial_commands_.SetDefaultHandler(cmd_unrecognized); 
} 
  
void update_serial_commands(void) 
{ 
  serial_commands_.ReadSerial(); // Appel au gestionnaire de commandes série 
}


Une erreur dans cette actualité ? Signalez-nous-la !