LOCODUINO

Réalisation de centrales DCC avec le logiciel libre DCC++

Réalisation de centrales DCC avec le logiciel libre DCC++ (1)

Comment adapter ce très bon logiciel à ses besoins propres

.
Par : Dominique

DIFFICULTÉ :

Nous avons découvert ce logiciel libre (merci Denis) en Mars 2016 par la visualisation de quelques vidéos très bien faites (si on comprend l’anglais) sur le DCC en général, et le projet en particulier.
En y regardant de plus près, la qualité de son écriture nous a amené à l’étudier en profondeur et à l’utiliser dans divers environnements matériels et applicatifs.

Les résultats sont excellents et les discussions et réalisations présentées sur le Forum Locoduino ici et par Tanguy, PAT95, msport, et bobyAndCo, méritent de rassembler l’essentiel du savoir-faire dans quelques articles, pour en faciliter l’utilisation et l’adaptation à vos propres projets.

Ce premier article, en français, est une présentation de ce qu’il offre comme possibilités au niveau de la partie "BaseStation" avec quelques exemples d’adaptations.

Dans les articles suivant nous aborderons le coté "Contrôleur" et nous présenterons des réalisations parmi lesquelles vous trouverez probablement tout ce qu’il faut pour vous lancer !

Article mis à jour en janvier 2022 : images agrandissables.

Présentation de DCC++

Le projet DCC++ a pour auteur un américain : Gregg E. Berman, qui y a travaillé depuis 2013 et jusqu’en 2016, avec la livraison, au printemps de cette année là, de sa dernière version sur GitHub.

C’est un très bon programmeur C++ à qui je rend hommage pour cet excellent projet (s’il lit ce fil, ce qui ne manquera pas d’arriver).

Attention : Une version sous forme de bibliothèque, plus pratique à utiliser que la version de Gregg est décrite dans l’article Bibliothèque DCCpp et se trouve sur la Forge de Locoduino.

Ce projet comprend deux parties :

  • une centrale DCC dite "BaseStation" qui se charge de la génération du courant de traction selon la norme NMRA DCC. Cette centrale est construite sur la base d’un Arduino UNO ou d’un MEGA, augmenté d’une carte "Motor Shield" pour la partie puissance (on verra plus loin comment élargir la base matérielle)
  • une application "Processing" dite "Controller" qui se charge du pilotage de la "BaseStation" soit via l’interface USB entre le PC et l’Arduino, soit via l’interface TCP/IP en ethernet ou wifi mais sur Mega seulement (on verra plus loin comment élargir les possibilités de "contrôle" notamment en HTML.
JPEG - 89.8 kio

Présentée comme cela, la BaseStation semble ne pas pouvoir se passer du Controller, comme une centrale SPROG a besoin d’un logiciel comme JMRI ou RocRail.
En réalité, d’une part, c’est vrai et il est possible de piloter la BaseStation par JMRI à partir de la version 4.1.16 ou RocRail (récent avec la bibliothèque dccpp) qui contiennent donc une interface spécifique pour DCC++.
Mais, d’autre part, ce n’est pas vrai, la centrale peut fonctionner de façon autonome à condition d’en modifier la programmation, ce qui fera l’objet d’exemples de réalisation décrits plus loin et dans les articles suivants.

Car Gregg a réalisé un logiciel Open Source admirable en C++ avec force commentaires et explications, et un découpage des fonctions en plusieurs onglets qui rendent la compréhension du logiciel beaucoup plus facile.

Voici ce qu’on obtient en quelques minutes en superposant un Arduino Mega, une carte Ethernet et une carte pour 2 moteurs (j’ai ajouté un radiateur sur le double pont en H L298 pour obtenir près de 2 A).

JPEG - 536.6 kio

Avec cet ensemble qui revient à 2 ou 3 dizaines d’Euros, on n’est pas très loin d’une Z21 ! Mais il faut maitriser les logiciels (il n’y a pas encore d’application pour tablette, mais ça finira par arriver :), donc avoir une certaine expérience. Je vais tenter de vous aider dans cette voie.

Pour ceux qui voudraient se documenter sur le Web, en langue américaine, voici les liens utiles qui vous diront tout :

  1. Le site de référence de DCC++
  2. Le Wiki sur GitHub
  3. Le téléchargement sur GitHub
  4. La page des commandes possibles de DCC++ via l’interface série ou ethernet
  5. Le Forum américain Trainboard avec ses 76 pages de discussions :
  6. Une page d’introduction sur Trainboard
  7. La page de support sur le site JMRI
  8. La page de support sur le site RocRail

Pour vous faciliter la tâche, le logiciel DCC++ a été recopié dans le GIT Locoduino.

Mais préférez plutôt DCCpp qui est ici sur le Git de Locoduino comme indiqué en début de cet article.

Il faut reconnaitre que Gregg à voulu particulièrement respecter la norme NMRA pour que sa centrale fournisse les meilleures caractéristiques :

  • adressage des locos sur 2 ou 4 octets (de 1 à 10239)
  • vitesses sur 128 crans
  • conduite simultanée de plusieurs locos (un dizaine)
  • commande des fonctions des locos (F0 à F28)
  • gestion des accessoires d’adresse 0 à 2048
  • programmation sur la voie principale, avec l’adresse DCC mais sans confirmation de la loco (écriture de CVs et de bits dans les CVs)
  • programmation sur la voie de programmation sans adresse DCC mais avec confirmation de la (seule) loco (lecture et écriture de CVs et de bits dans les CVs).

La centrale peut donc se connecter à la voie principale ET à la voie de programmation. C’est une configuration qui ressemble assez bien à la SPROG (sans vouloir les comparer).

J’ai rapidement découvert qu’on pouvait ajouter d’autres fonctions facilement comme expliqué dans ce fil du Forum : DCC++ est plein de promesses.

Dans la suite de cet exposé, je vais détailler d’abord le logiciel de la BaseStation en montrant comment on pourrait tirer partie des différents mécanismes qui le composent.

Il s’agit du sketch DCpp_Uno.

Ce sketch est accompagné de 17 onglets, certains toujours nécessaires, d’autres optionnels (ce qui ne veut pas dire que si on les enlève, ça va encore compiler, mais l’adaptation est facile à faire) :

D’abord les onglets nécessaires :

  • DCCpp_Uno.h contient les choix de configuration matériels principaux : le type d’Arduino (Uno ou Mega, -> mais en changeant quelques mots-clé, on peut faire tourner le logiciel aussi sur un Nano ou un Pro Mini s’ils utilisent tous un ATMega328 à 16Mhz, comme le démontre le fil http://forum.locoduino.org/index.ph...), le type de motor shield (Arduino ou Polulu, -> qui peut facilement être détourné vers un LMD18200 par exemple), et le type d’interface de communication (Série/USB ou Ethernet), le numéro de version (1.2.1+) et le mode diagnostic qui affiche les registres (non activé par défaut) qu’on va détailler plus loin.
  • Comm.h qui définit 3 types de cartes Ethernet si le choix "Ethernet" est retenu. On n’est plus très loin de la Z21 !
  • Config.h qui contient les réglages de la configuration choisie parmi celles possibles : le motor shield, le nombre maximum de registres (dont on va parler plus loin car c’est l’élément clé de DCC++), l’interface de communication, l’adresse IP, le port et l’adresse Mac (si Ethernet). Il est doc possible de connecter la BaseStation en Wifi avec une carte Ethernet Wifi si on le souhaite (je ne l’ai pas testé).
  • PacketRegister.h (les déclarations) et PacketRegister.cpp (le code) : c’est lui qui gère les contenus binaires des registres qui sont transmis bit à bit, répétitivement vers le circuit de puissance (motor shield) pour générer le signal DCC.
  • SerialCommand.h (les déclarations) et SerialCommand.cpp (le code) : c’est lui qui décode les commandes reçues sous forme de texte soit par la liaison série, soit par Ethernet. On trouvera la liste de ces commandes là : https://github.com/DccPlusPlus/Base....
  • CurrentMonitor.h (les déclarations) et CurrentMonitor.cpp (le code) : c’est lui qui gère la mesure de courant pour éviter les court-circuits, bien-sûr, mais surtout pour récupérer les réponses du décodeur en cours de programmation. C’est une fonction que je n’avais pas encore trouvée toute faite et ça tombe bien !

Ensuite les onglets dont on peut se passer éventuellement :

  • Accessories.h (les déclarations) et Accessories.cpp (le code) qui gère les commandes des accessoires
  • EEStore.h (les déclarations) et EEStore.cpp (le code) qui gère la sauvegarde des paramètres en EEPROM
  • Outputs.h (les déclarations) et Outputs.cpp (le code) qui pilote des sorties (commandes d’aiguille) de l’Arduino
  • Sensor.h (les déclarations) et Sensor.cpp (le code) qui gère des entrées (rétrosignalisation) de l’Arduino

L’architecture générale du logiciel DCC++

Cette collections d’onglets colle parfaitement à l’architecture du logiciel.

JPEG - 177.8 kio

La génération du signal DCC

C’est là où réside tout l’intérêt de ce logiciel :

Tout d’abord, il n’est pas inutile de rappeler le principe du DCC : transmettre un courant "alternatif" sur les rails, qui serve à la fois d’alimentation pour les trains et de transmission de commandes pour les décodeurs installés dans les locos.

Ce principe est illustré sur les figures suivantes :

PNG - 31.8 kio

Le bits "0" durent à peu près 2 fois plus longtemps que les bits "1".

PNG - 25.8 kio

Une commande DCC est une suite de "1" et de "0" qui sont, en fait, dans une suite d’octets pour chaque commande. Comme il faut pouvoir transmettre plusieurs commandes les unes après les autres et répéter cela sans arrêt, on place ces commandes dans des tableaux d’octets qui seront lus, bit par bit, par une tâche de fond automatique.

Dans le 1er article de la collection Comment piloter trains et accessoires en DCC avec un Arduino, j’explique comment créer des tableaux d’octets correspondant à des commandes DCC à émettre de façon répétitive par une routine d’interruption (donc en tâche de fond). Le plus compliqué, que je traite de façon probablement trop simplifiée dans cet article, consiste à remplir ces tableaux d’octets correctement et élégamment.

Et bien c’est ce que fait DCC++ de façon admirable !

DCC++ utilise donc des tableaux d’octets dits "Packet" organisés en "Register" qui sont gérés dans une liste "RegisterList". Ceci est décrit dans l’onglet PacketRegister.h. Les définitions de base sont :

struct Packet{
  byte buf[10];
  byte nBits;
}; // Packet

struct Register{
  Packet packet[2];
  Packet *activePacket;
  Packet *updatePacket;
  void initPackets();
}; // Register

struct RegisterList{  
  int maxNumRegs;
  Register *reg;
  Register **regMap;
  Register *currentReg;
  Register *maxLoadedReg;
  Register *nextReg;
  Packet  *tempPacket;
  byte currentBit;
  byte nRepeat;
  int *speedTable;
  int addressDccToDiscover;      // ajout Dominique
  static byte idlePacket[];
  static byte resetPacket[];
  static byte bitMask[];
  RegisterList(int);
  void loadPacket(int, byte *, int, int, int=0) volatile;
  void setThrottle(char *) volatile;
  void setFunction(char *) volatile;  
  void setAccessory(char *) volatile;
  void writeTextPacket(char *) volatile;
  void readCV(char *) volatile;
  void readCV_Main(char *s) volatile;      // ajout Dominique
  void writeCVByte(char *) volatile;
  void writeCVBit(char *) volatile;
  void writeCVByteMain(char *) volatile;
  void writeCVBitMain(char *s) volatile;  
  void printPacket(int, byte *, int, int) volatile;
};

Au passage, on voit dans RegisterList les méthodes disponibles dans DCC++, ainsi que celle que j’ai osé ajouter (readCV_Main). Mais nous y reviendrons plus loin.

Il peut y avoir 12 registres dans un Uno et probablement jusqu’à 50 dans un Mega. En tout cas c’est réglable dans la configuration (onglet Config.h).
Le registre N°0 est réservé pour les commandes à envoyer une seule fois (paquets Idle et Reset, commandes de programmation, commandes de fonctions, commandes d’accessoires). Les registres suivants N° 1 .. MAX_MAIN_REGISTERS sont réservés chacun à une machine sur le réseau. Il ne doit y avoir qu’un seul registre affecté à une même machine, sinon elle serait soumise à des ordres contradictoires et risque de s’agiter bizarrement.

Il y a un jeu de registres pour la voie principale et un autre, plus petit pour la voie de programmation car elles peuvent être commandées simultanément.

L’ensemble de ces Registres constitue donc un grand tableau de bits 0 et de bits 1 où chaque ligne est une trame DCC conforme à la norme NMRA et où chaque bit se traduit par une alternance de 200 microsecondes pour un bit ZERO ou 116 microsecondes pour un bit UN.

On se reportera à l’article L’Arduino et le système de commande numérique DCC pour se remémorer le principe de fonctionnement du bus DCC.

Pour produire le signal DCC automatiquement, en tâche de fond, il faut une routine sous interruption pilotée par les 2 compteurs OCR1A (pour les 2 alternances du bit) et OCR1B (pour chaque alternance, donc la moitié de la durée du bit) de l’ATMega. Il est évident que cette mécanique dépend du processeur utilisé et DCC++ ne s’applique dans son état actuel qu’au UNO et au MEGA. On peut évidemment le modifier légèrement pour qu’il fonctionne aussi sur NANO et PRO MINI car ils utilisent le même processeur que le UNO, un ATMega328. La modification serait plus importante sur le DUE mais la nécessité d’utiliser un DUE est très faible à mon avis, le MEGA étant largement suffisant.

Les 2 routines d’interruption, l’une pour la voie principale et l’autre pour la voie de programmation sont déclarées dans le programme principal DCCpp_Uno :

ISR(TIMER1_COMPB_vect){              // set interrupt service for OCR1B of TIMER-1 which flips direction bit of Motor Shield Channel A controlling Main Track
  DCC_SIGNAL(mainRegs,1)
}

#ifdef ARDUINO_AVR_UNO      // Configuration for UNO

ISR(TIMER0_COMPB_vect){              // set interrupt service for OCR1B of TIMER-0 which flips direction bit of Motor Shield Channel B controlling Prog Track
  DCC_SIGNAL(progRegs,0)
}

#else      // Configuration for MEGA

ISR(TIMER3_COMPB_vect){              // set interrupt service for OCR3B of TIMER-3 which flips direction bit of Motor Shield Channel B controlling Prog Track
  DCC_SIGNAL(progRegs,3)
}
#endif

Cette écriture simplifiée vient du fait que Gregg utilise une "Macro" qui évite d’écrire 3 fois à peu prês la même chose. J’avoue que je me suis gratté la tête en tombant là dessus, mais Thierry et Jean-Luc m’ont éclairé à temps.

Dans mon adaptation à une petite centrale dédiée à un va et vient, j’ai limité le fonctionnement à la seule voie principale et à un ATMega328 (un Nano, mais ce serait possible sur un Uno ou Pro Mini)

Dans le code qui suit, j’ai "dé-macro-isé" la routine pour la voie principale :

ISR(TIMER1_COMPB_vect){              // set interrupt service for OCR1B of TIMER-1 which flips direction bit of Motor Shield Channel A controlling Main Track
  //DCC_SIGNAL(mainRegs,1)
  if(mainRegs.currentBit==mainRegs.currentReg->activePacket->nBits){  /* IF no more bits in this DCC Packet */ 
    mainRegs.currentBit=0;                                            /*   reset current bit pointer and determine which Register and Packet to process next--- */    
    if(mainRegs.nRepeat>0 && mainRegs.currentReg==mainRegs.reg){      /*   IF current Register is first Register AND should be repeated */ 
      mainRegs.nRepeat--;                                             /*     decrement repeat count; result is this same Packet will be repeated */ 
    } else if(mainRegs.nextReg!=NULL){                                /*   ELSE IF another Register has been updated */ 
      mainRegs.currentReg=mainRegs.nextReg;                           /*     update currentReg to nextReg */ 
      mainRegs.nextReg=NULL;                                          /*     reset nextReg to NULL */ 
      mainRegs.tempPacket=mainRegs.currentReg->activePacket;          /*     flip active and update Packets */         
      mainRegs.currentReg->activePacket=mainRegs.currentReg->updatePacket; 
      mainRegs.currentReg->updatePacket=mainRegs.tempPacket;          
    } else{                                                           /*   ELSE simply move to next Register */ 
      if(mainRegs.currentReg==mainRegs.maxLoadedReg)                  /*     BUT IF this is last Register loaded */ 
        mainRegs.currentReg=mainRegs.reg;                             /*       first reset currentReg to base Register, THEN */ 
      mainRegs.currentReg++;                                          /*     increment current Register (note this logic causes Register[0] to be skipped when simply cycling through all Registers) */ 
    }                                                                 /*   END-ELSE */ 
  }                                                                   /* END-IF: currentReg, activePacket, and currentBit should now be properly set to point to next DCC bit */ 
                                                                      
  if(mainRegs.currentReg->activePacket->buf[mainRegs.currentBit/8] & mainRegs.bitMask[mainRegs.currentBit%8]){     /* IF bit is a ONE */ 
    OCR1A=DCC_ONE_BIT_TOTAL_DURATION_TIMER1;                          /*   set OCRA for timer N to full cycle duration of DCC ONE bit */ 
    OCR1B=DCC_ONE_BIT_PULSE_DURATION_TIMER1;                          /*   set OCRB for timer N to half cycle duration of DCC ONE but */ 
  } else{                                                             /* ELSE it is a ZERO */ 
    OCR1A=DCC_ZERO_BIT_TOTAL_DURATION_TIMER1;                         /*   set OCRA for timer N to full cycle duration of DCC ZERO bit */ 
    OCR1B=DCC_ZERO_BIT_PULSE_DURATION_TIMER1;                         /*   set OCRB for timer N to half cycle duration of DCC ZERO bit */ 
  }                                                                   /* END-ELSE */  
                                                                       
  mainRegs.currentBit++;                                              /* point to next bit in current Packet */  
}

Rien qu’avec cette routine d’interruption, et après l’initialisation des variables qui doit être faite dans le Setup(), notre BaseStation est capable d’emblée de produire le signal d’alimentation DCC des voies.

Il reste à comprendre maintenant comment les Registres sont programmés (octets et bits), c’est à dire comment BaseStation peut envoyer des commandes DCC sur les rails.

Pour ce faire, il y a une méthode de RegisterList qui est très importante et qu’il ne faut absolument pas modifier : c’est la méthode loadPacket

// LOAD DCC PACKET INTO TEMPORARY REGISTER 0, OR PERMANENT REGISTERS 1 THROUGH DCC_PACKET_QUEUE_MAX (INCLUSIVE)
// CONVERTS 2, 3, 4, OR 5 BYTES INTO A DCC BIT STREAM WITH PREAMBLE, CHECKSUM, AND PROPER BYTE SEPARATORS
// BITSTREAM IS STORED IN UP TO A 10-BYTE ARRAY (USING AT MOST 76 OF 80 BITS)

void RegisterList::loadPacket(int nReg, byte *b, int nBytes, int nRepeat, int printFlag) volatile {
  
  nReg=nReg%((maxNumRegs+1));         // force nReg to be between 0 and maxNumRegs, inclusive

  while(nextReg!=NULL);               // pause while there is a Register already waiting to be updated -- nextReg will be reset to NULL by interrupt when prior Register updated fully processed
 
  if(regMap[nReg]==NULL)              // first time this Register Number has been called
   regMap[nReg]=maxLoadedReg+1;       // set Register Pointer for this Register Number to next available Register
 
  Register *r=regMap[nReg];           // set Register to be updated
  Packet *p=r->updatePacket;          // set Packet in the Register to be updated
  byte *buf=p->buf;                   // set byte buffer in the Packet to be updated
          
  b[nBytes]=b[0];                     // copy first byte into what will become the checksum byte  
  for(int i=1;i<nBytes;i++)           // XOR remaining bytes into checksum byte
    b[nBytes]^=b[i];
  nBytes++;                           // increment number of bytes in packet to include checksum byte
      
  buf[0]=0xFF;                        // first 8 bytes of 22-byte preamble
  buf[1]=0xFF;                        // second 8 bytes of 22-byte preamble
  buf[2]=0xFC + bitRead(b[0],7);      // last 6 bytes of 22-byte preamble + data start bit + b[0], bit 7
  buf[3]=b[0]<<1;                     // b[0], bits 6-0 + data start bit
  buf[4]=b[1];                        // b[1], all bits
  buf[5]=b[2]>>1;                     // b[2], bits 7-1
  buf[6]=b[2]<<7;                     // b[2], bit 0
  
  if(nBytes==3){
    p->nBits=49;
  } else{
    buf[6]+=b[3]>>2;                  // b[3], bits 7-2
    buf[7]=b[3]<<6;                   // b[3], bit 1-0
    if(nBytes==4){
      p->nBits=58;
    } else{
      buf[7]+=b[4]>>3;                // b[4], bits 7-3
      buf[8]=b[4]<<5;                 // b[4], bits 2-0
      if(nBytes==5){
        p->nBits=67;
      } else{
        buf[8]+=b[5]>>4;              // b[5], bits 7-4
        buf[9]=b[5]<<4;               // b[5], bits 3-0
        p->nBits=76;
      } // >5 bytes
    } // >4 bytes
  } // >3 bytes
  
  nextReg=r;
  this->nRepeat=nRepeat;
  maxLoadedReg=max(maxLoadedReg,nextReg);
  
  if(printFlag && SHOW_PACKETS)       // for debugging purposes
    printPacket(nReg,b,nBytes,nRepeat);  

} // RegisterList::loadPacket

C’est loadPacket qui va positionner les octets et les bits de chaque commande DCC dans un des registres.

LoadPacket contient aussi un mécanisme très important : une sorte de synchronisation avec la routine d’interruption ISR (pendant laquelle le programme principal est mis en attente) qui empêche la modification d’un registre tant que l’appel précédent de loadPacket n’a pas complètement terminé sont travail et que la routine ISR n’a pas envoyé le paquet DCC sur les rails.

C’est la ligne suivant qui fait cela :

while(nextReg!=NULL);               // pause while there is a Register already waiting to be updated -- nextReg will be reset to NULL by interrupt when prior Register updated fully processed

Cette astuce permet de simplifier considérablement l’écriture du programme dans les niveaux plus hauts en faisant abstraction des contraintes précitées.

Par exemple, dans une autre fonction de RegisterList, on trouve :

loadPacket(0,resetPacket,2,3);          // NMRA recommends starting with 3 reset packets
loadPacket(0,bRead,3,5);                // NMRA recommends 5 verify packets
loadPacket(0,resetPacket,2,1);          // forces code to wait until all repeats of bRead are completed (and decoder begins to respond)

On est certain, grâce à ce mécanisme, que les paquets seront bien transmis intégralement sur les rails dans l’ordre indiqué. C’est génial, moi j’aime !

Avant de conclure sur cette partie, voici les autres fonctions présentes dans RegisterList. Leurs arguments sont contenus dans un tableau de caractères :

  void setThrottle(char *); // commande de vitesse (0..126) et directions (1=avant, 0=arriere)
  void setFunction(char *); // commande de fonction loco F0..F28
  void setAccessory(char *); // commande d'accessoire (0..2048)
  void writeTextPacket(char *); //envoi d'un paquet libre
  void readCV(char *); // lecture d'un CV sur voie de programation
  void readCV_Main(char *s); // lecture d'un CV sur voie principale (ajout perso de Dominique)
  void writeCVByte(char *); // écriture d'un octet entier de CV sur voie de programation
  void writeCVBit(char *); // écriture d'un bit de CV sur voie de programation
  void writeCVByteMain(char *); // écriture d'un octet entier de CV sur voie principale (sans réponse du décodeur)
  void writeCVBitMain(char *s);  // écriture d'un bit de CV sur voie principale (sans réponse du décodeur)

Si vous comparez avec le source original de BaseStation, vous trouverez que la fonction readCV_Main(char *s) n’existe pas. C’est un exemple de modification qui est facile à faire. Dans cet exemple, j’ai simplement adressé la commande aux registres de la voie principale et j’ai pris les mesures de courant sur la voie principale (en lieu et place de la voie de programmation). Il va de soi que cette fonction n’a d’intérêt que si une seule loco est présente sur la voie principale.

J’ai imaginé cette modification dans le but de récupérer l’adresse DCC de la loco posée sur les rails pour avoir un automatisme total de va et vient.

L’interface avec le Contrôleur par liaison série ou ethernet

Nous avons vu dans l’épisode précédent comment la BaseStation fabrique ses paquets DCC à partir de commandes sous forme textuelle qui sont passées à des fonctions de l’objet RegisterList (qui n’est pas une classe mais une structure, mais bon..).

Les onglets SerialCommand.h et SerialCommand.cpp vont se charger d’assurer l’interface entre le canal de communication (série ou ethernet) et les fonctions de RegisterList.

SerialCommand.h déclare une structure SerialCommand :

struct SerialCommand{
  static char commandString[MAX_COMMAND_LENGTH+1];
  static volatile RegisterList *mRegs, *pRegs;
  static CurrentMonitor *mMonitor;
  static void init(volatile RegisterList *, volatile RegisterList *, CurrentMonitor *);
  static void parse(char *);
  static void process();
}; // SerialCommand

On y trouve pas mal de choses importantes :

  • Un tableau de caractères commandString : c’est la commande
  • Deux RegisterList, l’une mRegs pour la voie principale et l’autre pRegs pour la voie de programmation. C’est ce qui permet d’appeler simplement les fonctions de RegisterList vues précédemment
  • Une structure CurrentMonitor qui permet d’avoir une mesure de courant
  • Une fonction parse qui se chargera du décodage des commandString et qui va donc appeler les fonctions de RegisterList
  • Une fonction process qui va lire l’interface série ou ethernet et va passer la commande reçue à la fonction parse

Le code d’exécution se trouve donc dans l’onglet SerialCommand.cpp :

La fonction process ne fait pas grand chose d’autre que d’enregistrer les caractères compris entre "<" et ">" et de passer le résultat à parse.

La fonction parse analyse le 1er caractère de la commande pour envoyer le reste des caractères (sans cette 1ère lettre) vers la bonne fonction de RegisterList

"t" pour setThrottle (commande DCC)
"f" pour setFunction (commande DCC)
"a" pour setAccessory (commande DCC)
"T" pour une commande d’aiguille directe (commande non DCC)
"Z" pour une commande d’une pin de sortie de l’Arduino (commande non DCC)
"S" pour lire l’état d’un capteur relié à une pin d’entrée de l’Arduino (commande non DCC)
"Q" pour lire l’état de tous les capteurs à la fois (commande non DCC)
"w" pour programmer un CV sur la voie principale en écriture seule sans vérification (commande DCC)
"b" pour programmer un bit particulier d’un CV sur la voie principale en écriture seule sans vérification (commande DCC)
"W" pour programmer un CV sur la voie de programmation avec vérification donc réponse du décodeur (commande DCC)
"B" pour programmer un bit particulier d’un CV sur la voie de programmation avec vérification donc réponse du décodeur (commande DCC)
"R" pour lire un CV sur la voie de programmation avec réponse du décodeur évidemment (commande DCC)
"1" pour appliquer le signal DCC sur les rails (power ON)
"0" pour couper le signal DCC sur les rails (power OFF)
"c" pour lire la valeur du courant sur la voie principale (commande non DCC)
"s" pour envoyer sur la liaison un état du système (commande non DCC)
"E" pour sauvegarder les valeurs des aiguilles et capteurs en EEPROM (commande non DCC)
"e" pour effacer les valeurs enregistrées dans l’EEPROM (commande non DCC)

A cela s’ajoute quelques utilitaires comme :
"M" pour envoyer un paquet DCC de 2 à 5 octets sur la voie principale (commande DCC)
"P" pour envoyer un paquet DCC de 2 à 5 octets sur la voie de programmation (commande DCC)

Si on s’arrête un instant pour regarder la loop() dans DCCp_Uno, on y lit :

///////////////////////////////////////////////////////////////////////////////
// MAIN ARDUINO LOOP
///////////////////////////////////////////////////////////////////////////////

void loop(){
  
  SerialCommand::process();              // check for, and process, and new serial commands
  
  if(CurrentMonitor::checkTime()){      // if sufficient time has elapsed since last update, check current draw on Main and Program Tracks 
    mainMonitor.check();
    progMonitor.check();
  }

  Sensor::check();    // check sensors for activate/de-activate
  
} // loop

Ce qui veut dire que le programme BaseStation ne fait rien d’autre que de traiter les commandes ci-dessus venues de la liaison série ou ethernet, ainsi que de surveiller le courant en cas de court-circuit et de tester les capteurs, ce qui se traduit par l’envoi de messages vers la liaison série ou ethernet.

Il ne fait donc rien de lui-même, mais cela nous ouvre la voie vers des réalisations personnelles intéressantes et sophistiquées, avec une telle belle boite à outils.

Mais revenons à la fonction parse :

On constate qu’il y a deux familles de fonctions :
Celles qui sont conformes à la norme DCC NMRA : t, f, a, w, b, W, B, R, M et P
Celles qui sont spécifiques à BaseStation et ne concernent pas la norme DCC : T, Z, S, Q, 1, 0, c, s, E, e

Les fonctions conformes à la norme DCC sont réalisées par RegisterList comme on l’a vu précédemment. Mais RegisterList ne s’occupe que des fonctions propres au DCC puisqu’il se charge de la gestion des paquets DCC envoyés aux rails.

Alors, pour les autres fonctions, il existe des onglets spécifiques qui traitent des fonctions non DCC :

Les onglets Accessories.h et Accessories.cpp s’occupent des aiguilles en commande directe par l’Arduino (et non en DCC via un décodeur d’accessoire, ce que BaseStation supporte aussi via la fonction setAccessory). Elles gèrent les structures :

struct TurnoutData {
  byte tStatus;
  byte subAddress;
  int id;
  int address;  
};

struct Turnout{
  static Turnout *firstTurnout;
  int num;
  struct TurnoutData data;
  Turnout *nextTurnout;
  void activate(int s);
  static void parse(char *c);
  static Turnout* get(int);
  static void remove(int);
  static void load();
  static void store();
  static Turnout *create(int, int, int, int=0);
  static void show(int=0);
}; // Turnout

On y retrouve une fonction parse qui fait le lien avec la fonction parse de SerialCommand, pour permettre au contrôleur de commander les aiguilles de cette façon.

Les onglets Outputs.h et Outputs.cpp s’occupent des pins disponibles de l’Arduino qui peuvent être programmées en sortie. Elles gère les structures :

struct OutputData {
  byte oStatus;
  int id;
  byte pin; 
  byte iFlag; 
};

struct Output{
  static Output *firstOutput;
  int num;
  struct OutputData data;
  Output *nextOutput;
  void activate(int s);
  static void parse(char *c);
  static Output* get(int);
  static void remove(int);
  static void load();
  static void store();
  static Output *create(int, int, int, int=0);
  static void show(int=0);
}; // Output

On y retrouve aussi une fonction parse qui fait le lien avec la fonction parse de SerialCommand, pour permettre au contrôleur de commander les pins de sortie de l’Arduino.

Les onglets Sensor.h et Sensor.cpp s’occupent des pins disponibles de l’Arduino qui peuvent être programmées en entrée pour lire des états de capteurs. Elles gèrent les structures :

struct SensorData {
  int snum;
  byte pin;
  byte pullUp;
};

struct Sensor{
  static Sensor *firstSensor;
  SensorData data;
  boolean active;
  float signal;
  Sensor *nextSensor;
  static void load();
  static void store();
  static Sensor *create(int, int, int, int=0);
  static Sensor* get(int);  
  static void remove(int);  
  static void show();
  static void status();
  static void parse(char *c);
  static void check();   
}; // Sensor

On y retrouve encore une fonction parse qui fait le lien avec la fonction parse de SerialCommand, pour permettre au contrôleur de lire les états des capteurs de rétrosignalisation.

De plus, la fonction check est scrutée par la loop de façon à informer le contrôleur au plus vite des changements d’état des capteurs.

Bien entendu je ne vais pas rentrer dans l’analyse des fonctions de ces 3 dernières familles, mais une série de conclusions d’imposent :

1) BaseStation est un logiciel très complet rédigé de façon propre et claire donc fiable (bien que je n’ai pas tout testé, loin s’en faut, mais cela fleure bon).

2) Sa modularité en fait un logiciel adaptable : on peut enlever et ajouter des fonctions, on peut réaliser certaines fonctions différemment, mais il convient de respecter le noyau principal qui est le moteur DCC.

3) BaseStation est principalement destiné à être piloté par un logiciel de contrôle sur PC. L’auteur Gregg a écrit DCC++Controleur en Processing (version récente) : on peut l’apprécier ou pas. D’autres ont réalisé les adaptations dans JMRI et RocRail (dernières versions seulement).
Presonnellement, j’ai réalisé une petite manette minimaliste en Processing, en une heure de débutant :

PNG - 99.8 kio

Le fond est rouge quand le DCC n’est pas envoyé sur les rails, vert dans le cas contraire. Des images représentent l’état des feux, du sens de déplacement, ...
Vous pouvez télécharger cet exemple sur le fil du Forum DCC++ sur Nano avec LMD18200

Et on trouvera bientôt dans Locoduino des réalisations de serveurs Web pour piloter DCC++ via votre réseau local.

4) Mais rien n’empêche d’ajouter d’autres fonctions, notamment un automate de gestion de circulation, une gestion de Bal, une gestion de signaux, etc..
Les modèles de programmation des onglets peuvent être une bonne base pour ces extensions. Un bel automate peut aussi trouver sa place dans la loop ou appelé depuis la loop.

La détection de courant de DCC++

Un des gros avantages de DCC++ par rapport aux autres logiciels connus qui génèrent du DCC est sa mesure de courant qui fonctionne bien sur divers matériels (shield Arduino Motor, ou Polulu ou LMD18200).

La mesure de courant dans la LOOP (protection contre les court-circuits)

La mesure de courant se fait par la lecture d’un port analogique à chaque tour de loop() et l’intérêt de cette mesure est qu’elle intègre une moyenne glissante ou "lissage exponentiel" avec les dernières mesures, c’est une sorte de filtre passe-bas : on n’est pas tous capable d’inventer cela, donc profitons en ;)

La formule utilisée est c_t = coeff x apin + c_t-1 x (1-coeff) où

  • c_t est la valeur de courant calculée au tour t
  • c_t-1 est la valeur de courant calculée au tour t-1 (précédent)
  • apin est la valeur analogique lue sur la pin analogique utilisée pour la mesure
  • coeff est le coefficient de pondération qui est ici de 1%

La valeur lue sur le port apin est donc réduite au 1/100 ème de sa valeur et ajoutée à 99/100 du résultat précédent (initialisé à 0 au départ). La valeur calculée converge donc progressivement vers une valeur moyenne du courant, avec un grande stabilité car les mesures affectent peu le résultat.

On trouve le code suivant dans l’onglet CurrentMonitor.cpp

void CurrentMonitor::check(){
#define  CURRENT_SAMPLE_SMOOTHING   0.01

  current=analogRead(pin)*CURRENT_SAMPLE_SMOOTHING+current*(1.0-CURRENT_SAMPLE_SMOOTHING);        // compute new exponentially-smoothed current
  if(current>CURRENT_SAMPLE_MAX && digitalRead(SIGNAL_ENABLE_PIN_PROG)==HIGH){                    // current overload and Prog Signal is on (or could have checked Main Signal, since both are always on or off together)
    digitalWrite(SIGNAL_ENABLE_PIN_PROG,LOW);                                                     // disable both Motor Shield Channels
    digitalWrite(SIGNAL_ENABLE_PIN_MAIN,LOW);                                                     // regardless of which caused current overload
    INTERFACE.print(msg);                                                                         // print corresponding error message
  }    
} // CurrentMonitor::check  

Ce code, appelé dans la loop permet de couper l’alimentation en cas de dépassement de seuil (ici 300), c’est à dire en cas de court-circuit.

La mesure de courant pour les commandes de programmation

C’est l’application la plus interessante de cette mesure de courant.

Nous avons tous remarqué que lors de la programmation du décodeur d’une loco (on commence toujours par son adresse DCC), celle-ci se met à bouger :-[
Cela vient du fait que le décodeur va provoquer des impulsions de consommation de courant sur les rails en activant le moteur pendant une fraction de seconde. Il répète plusieurs cycles de "consommation" pour transmettre une information qui est composée de bits 0 et de bits 1.

Pour récupérer cette information, notre logiciel BaseStation va d’abord mesurer une valeur de repos du courant : "base" c’est à dire quand la loco est au repos : ce courant n’est pas nul car le décodeur en consomme toujours un peu.
Puis quand la loco va transmettre ses "bits" le logiciel va mesurer la consommation pendant la durée du bit, rechercher le pic de consommation (la valeur la plus élévée pendant ce laps de temps) puis comparer le résultat avec la valeur de repos. Si le résultat dépasse la valeur de repos d’un seuil ACK_SAMPLE_THRESHOLD alors le bit est un "1", sinon c’est un "0"

Ce mécanisme se répète 8 fois pour récupérer les 8 bits d’un octet de réponse de la loco.

Les constantes utilisées dans cette mesure se trouvent dans l’onglet PacketRegister.h :

// Define constants used for reading CVs from the Programming Track

#define  ACK_BASE_COUNT            100      // number of analogRead samples to take before each CV verify to establish a baseline current
#define  ACK_SAMPLE_COUNT          500      // number of analogRead samples to take when monitoring current after a CV verify (bit or byte) has been sent 
#define  ACK_SAMPLE_SMOOTHING      0.2      // exponential smoothing to use in processing the analogRead samples after a CV verify (bit or byte) has been sent
#define  ACK_SAMPLE_THRESHOLD       30      // the threshold that the exponentially-smoothed analogRead samples (after subtracting the baseline current) must cross to establish ACKNOWLEDGEMENT

On constate que les paramètres de lissage ne sont pas tout à fait les même que pour la mesure dans la loop. Cela vient du fait que les mesures dans la loop se font en fonction du temps d’éxécution de la loop toute entière, alors que dans la réponse lors d’une programmation, ce sont seulement quelques instructions qui s’éxécutent.

Et le code de récupération de chaque bit se trouve dans l’onglet PacketRegister.cpp, à l’intérieur de chaque fonction de programmation avec réponse, comme readCV, writeCVByte, writeCVBit sur la voie de programmation :

    c=0;
    d=0;
    base=0;

    for(int j=0;j<ACK_BASE_COUNT;j++)
      base+=analogRead(CURRENT_MONITOR_PIN_PROG);
    base/=ACK_BASE_COUNT;

    bRead[2]=0xE8+i;  

    loadPacket(0,resetPacket,2,3);          // NMRA recommends starting with 3 reset packets
    loadPacket(0,bRead,3,5);                // NMRA recommends 5 verfy packets
    loadPacket(0,resetPacket,2,1);          // forces code to wait until all repeats of bRead are completed (and decoder begins to respond)

    for(int j=0;j<ACK_SAMPLE_COUNT;j++){
      c=(analogRead(CURRENT_MONITOR_PIN_PROG)-base)*ACK_SAMPLE_SMOOTHING+c*(1.0-ACK_SAMPLE_SMOOTHING);
      if(c>ACK_SAMPLE_THRESHOLD)
        d=1;
    }

Il va de soi que ce code ne doit pas être modifié car il fonctionne bien : nous avons testé ce code sur plusieurs plateformes à base d’Uno, Nano et Mega.

Avec ce mécanisme, j’ai réalisé quelque chose qui ne semble pas exister jusqu’à présent : une demande automatique du numéro de CV d’un loco qui serait seule sur une voie unique, notamment pour automatiser un va et vient.

Le principe est le suivant : je teste le bit 5 du cv#29. S’il est égal à 1, c’est une adresse longue codée dans les CV 17 et 18, sinon, c’est une adresse courte codée dans le CV 1.

Je vous donne la méthode que j’ai écrite pour cela :

int trouveAdresseDCC() {
  int  areponse = -1;
  SerialCommand::getCV29(); 
  areponse = mainRegs.valeurCV;
  if ((areponse != -1) && (bitRead(areponse, 5))) {
    // adresse longue : lire CV#17 et CV#18
    SerialCommand::getCV18(); 
    areponse = mainRegs.valeurCV;
    if (areponse != -1) {
      SerialCommand::getCV17(); 
      if (mainRegs.valeurCV != -1) {
        areponse = areponse + ((mainRegs.valeurCV - 192)<<8);
      }
    }
  } else {
    // adresse courte : lire CV#1
    SerialCommand::getCV1(); 
    areponse = mainRegs.valeurCV;
  }
  return(areponse);
}

Et je vous donne aussi les commandes GetCVxx() que j’ai ajoutées dans l’onglet PacketRegsiter :

void SerialCommand::getCV29() {
  //char commandStr[14] = "r 29 129 129 ";
  //commandStr[12] = '\0';
  //parse(commandStr);  
  char commandStr[12] = "29 129 129 ";
  mRegs->readCV_Main(commandStr);
}

void SerialCommand::getCV1() {
  //char commandStr[12] = "r 1 101 101";
  //commandStr[11] = '\0';
  //parse(commandStr);  
  char commandStr[12] = "1 101 101 ";
  mRegs->readCV_Main(commandStr);
}

void SerialCommand::getCV17() {
  //char commandStr[14] = "r 17 117 117";
  //commandStr[12] = '\0';
  //parse(commandStr);  
  char commandStr[12] = "17 117 117 ";
  mRegs->readCV_Main(commandStr);
}

void SerialCommand::getCV18() {
  //char commandStr[14] = "r 18 118 118";
  //commandStr[12] = '\0';
  //parse(commandStr); 
  char commandStr[12] = "18 118 118 ";
  mRegs->readCV_Main(commandStr); 
}

Ceci pour vous montrer à quel point on peut tirer profit de DCC++ à chacun sa manière.

Comme, par exemple, en réalisant une centrale minimal à base de Nano :

JPEG - 114.7 kio

dont vous pouvez suivre la réalisation sur le fil du Forum DCC++ sur Nano avec LMD18200.

Mais ce n’est pas tout ...

Cet article vous laisse entrevoir de nombreuses possibilités de pilotage et de gestion de votre réseau, sans l’achat d’une centrale du marché, je l’espère.

Nous avons vu un exemple de modification de "l’interieur". Celui-ci sera mis en oeuvre prochainement dans une nouvelle version de mon va-et vient.

Mais nous verrons aussi bientôt un exemple de pilotage par l’interface Ethernet et les immenses possibilités que cela permet notamment pour gérer un parc de locos.

A bientôt donc...

BaseStation confirme donc son universalité en permettant de commander la voie principale, et aussi la voie de programmation avec un compatibilité NMRA bien étudiée.

Les suites de cet article seront consacrées à plusieurs projets :

  • la commande par Ethernet et la réalisation d’un mini serveur Internet pour piloter un parc de machine à partir de son PC ou sa tablette.
  • la réalisation d’une nouvelle version de mon va-et-vient automatique qui marche à merveille !

... et j’espère aussi : vos propres réalisations.

A suivre...

36 Messages

  • Bonjour,

    Quelques détails utiles. D’abord sur ces motor shields sur base L298, pour ne pas les cramer, et pour ne pas cramer un Arduino.

    Avec le L298 et ces shields, il ne faut probablement pas monter à beaucoup plus d’un ampère (voie de programmation et tests, ou voie, mais avec peu de décodeurs et matériel roulant) en surveillant la température du L298. Certains de ces shields dont le modèle que vous présentez ont un bug thermique. Le dissipateur du boitier L298/PowerSO20 est dessous ce composant. Un dissipateur par dessus ce boitier n’est pas la solution pour monter à 2A, le L298 pourrait rapidement surchauffer puis cramer. En théorie, la chaleur est à évacuer via le PCB, qui n’est pas prévu pour, il y manque les plans de cuivre évacuant la chaleur du L298. De surcroit, sur certains lots de ces shields, le L298 n’est même pas plaqué contre le PCB ; on peu alors glisser une feuille de papier entre le composant et le PCB, ce qui n’améliore en rien la dissipation thermique du PowerSO20 via le PCB. C’est discuté sur les forums anglais, s’y référer pour les détails.

    Les Arduino ne tolèrent théoriquement pas plus de 12V sur VIN. Or pour du HO, une alim pour le shield et les voies sera à 15V. Il sera donc préférable de couper VIN du motor shield (une piste VIN-CONNECT est à couper) afin d’avoir un VIN du shield (15V) différent du VIN de l’Arduino (12V max, sinon 5V via USB).

    Le shield L298 peut être utile aux tests ou avec peu de matériel DCC, sous 12V. Le LMD18200 avec un bon dissipateur serait mieux pour des courants de 2A et plus, à plus de 12V.

    Attention aux longueurs des pattes des shields, il y en a de courtes. Qu’il n’y ait pas de court-circuits une fois les shields ethernet/motor empilés sur l’Arduino.

    Sur l’utilisation et les débuts avec DCC++

    Dans un des fichiers .h on peut définir le type de motor shield qu’on utilise, cela adapte les I/O Arduino au modèle de shield. Dans un de ces fichiers .h on peut également retrouver les I/O Arduino selon le type de shield, et repérer ainsi quelle est la sortie du signal DCC. On peut alors même se passer d’un shield pour de premiers tests avec uniquement le code de DCC++ et un signal DCC niveau TTL/5V.

    Car il suffit ensuite d’un Arduino "A" avec DCC++ connecté directement, par fil, sans booster ni optocoupleur, vers un second Arduino, "B". Une dizaines d’Euros, et on peut débuter avec le DCC, étudier les sources de DCC++ sur un UNO avec les sources de décodeurs flashés sur mini ou micro. UNO /DCC++ pinX /DCC out <-> micro INTy /DCCin.

    Pour DCC++, un Arduino Mega ne semble être utile que pour plus d’EEPROM (stocker les états d’aiguillages), pour des I/O en plus (détecteurs de présence...), ainsi que pour plus de flash et RAM (ethernet shield et évolutions/personnalisation du code). Pour débuter, via USB, sans ethernet, un simple UNO va suffire pour y flasher DCC++.

    Puis enfin... dans les sources de DCC++, il m’avait semblé que le code est aujourd’hui adapté au module ethernet de type W5200 (avec Mega uniquement).

    Cordialement

    Répondre

  • Merci pour toutes ces précisions utiles.

    Si vous voulez avoir beaucoup d’Ampères, il existe une carte à base de BTS7960B qui peut monter jusqu’à 40A, de quoi alimenter la totalité de MiniWorld Lyon !!
    Mais j’avoue que je ne l’ai pas testée, bien que j’en ai acheté une pour voir un de ces jours !

    Certes, la carte moteurs Arduino (L298) est limitée en puissance, ce qui est très bien expliqué (en américain) dans le Forum américain Trainboard où ce sujet est très bien développé.

    Je vous encourage à le lire : il y a plein d’autres renseignements utiles qui pourraient compléter nos articles en français dans l’avenir.

    Ayant un réseau en N, je l’ai testé sans problème avec une seule loco. C’est suffisant pour une voie de programmation.

    Mais la carte à base de LMD18200 n’est pas qu’une "mode" sur Locoduino, ce circuit ayant des performances bien supérieures aux L298 (et autres similaires), si vous regardez ses caractéristiques.
    C’est un composant ESSENTIEL dans le réseau et il ne faut pas chipoter pour quelques Euros de plus (tellement peu en fait !).

    Pour l’Ethernet, patience, ça va venir ;)

    Cordialement

    Répondre

  • De rien. Pour des questions de sécurité, je pense qu’il faut préférer des solutions L298 (2A max s’il est bien refroidi, pour un mini réseau) ou avec LMD18200 (3A continu, crête à 5A, protection interne/surge à 10A, selon la doc du LMD18200), puis multiplier les boosters de type LMD18200, de petite dimension.
    Car avec un booster susceptible de délivrer un courant très fort, jusqu’à 40A crête (c’est possible, selon l’alim en amont), au moindre court-circuit tel que dans un aiguillage, des contacteurs (micro switch et autres) ainsi que les appareils et les roues pourraient être marqués voire détruits par un arc électrique.
    Je compte utiliser DCC++ avec un L298 en faisant suivre plusieurs LMD18200 pour alimenter des voies, des portions de voies. Quelle que soit la solution que je retiendrais, je m’assurerais d’abord qu’une protection électronique et rapide limite efficacement le courant de court-circuit à 8-10A, et même à moins que cela.
    Votre booster à 40A, mieux que L298 ou LMD18200, pourrait être associé à un bloc régulateur 15V en amont, qui en limiterait le courant de court-circuit à 5, 8 ou 10A grand maximum.
    Cordialement

    Répondre

  • « Votre booster à 40A, mieux que L298 ou LMD18200, pourrait être associé à un bloc régulateur 15V en amont, qui en limiterait le courant de court-circuit à 5, 8 ou 10A grand maximum. »

    Là je blaguais :))

    Ce circuit à base de BTS7960B est plutôt fait pour les trains de jardin et encore, pour des grosses locos qui tirent beaucoup de wagons !

    Je ne voudrais pas qu’il serve en intérieur : au moindre court-circuit, le risque d’incendie est probable si la protection n’est pas garantie.

    Pour le reste, d’accord.
    Mais c’est intéressant de suivre ce que vous allez faire pour asservir des LMD18200 sur des L298.
    Pourquoi ne pas mettre un LMD18200 en tête puisqu’il a des fronts de montée/descente plus rapides que le L298 ?

    Répondre

  • Le BTS7960B pourrait servir à ceux qui voudront réaliser un booter unique à 5, 8 ou 10A (avec les risques que cela comporte pour les appareils, roues et switchs).

    > Je ne voudrais pas qu’il serve en intérieur
    C’est pourquoi je mettais moi-même en garde sur ces courants potentiellement très forts et en parlant d’une limitation électronique efficace en courant. Les centrales peuvent détecter un court-circuit, a/ si le soft n’est pas planté b/ s’il n’est pas déjà trop tard.

    Pour le moment, je me familiarise avec les briques, dont ce L298 éco et facile à utiliser. Puis j’ajouterais plus puissant. Je n’envisage pas d’utiliser un LMD18200 et une grosse alim en tête pour éviter des risques tels que de courts circuits (puis d’incendie aussi). J’y viendrais peut-être, pour des fronts plus rapides, si c’est utile ; mais en ce cas, pas sans une protection en amont d’un LMD18200 en tête. Car le LMD18200 ne "disjoncte" qu’à 10A...

    opendcc (leur site) propose des architectures avec des bus tel que HS88, HS88-n ou BiDi. Ces archi en bus intègrent le signal DCC et permettent d’y raccorder aussi les "vrai" boosters, ceux qui alimentent les voies. C’est ce genre de modèle que je compte suivre afin de distinguer mes câblages courants faibles pour DCC/contrôle et feedback de câblages pour courants plus forts.

    Avec une telle architecture (centrale, bus, puis boosters) et un petit réseau, un L298 sur la centrale pourrait suffire.

    Cordialement

    Répondre

    • 40 A en 15 ou 18 V, pas besoin de prévoir d’éclairage complémentaire, les étincelles suffiront :-)))))

      Je n’entends pas parler du shield Pololu MC33926 qui fait pourtant partie de hypothèses proposées dans DCC++. Personnellement, j’ai commencé en DCC++ avec un LMD18200 (en HO), solution qui m’a donné totale satisfaction pendant tous les tests. Par contre par la suite, j’ai opté pour le Pololu tout simplement parcequ’il inclus deux sorties, une pour la voie principale + une pour la voie de programmation (Il faudrait sinon 2 LMD). La détection de court-circuit est incluse (2 MAX471 sinon). Le Pololu est donné pour 3A. En réalité, sur des pics comme la mise sous tension du circuit avec 4 ou 5 locos bien sûr à l’arrêt, le Pololu se court circuite. Par contre, en utilisation normale, il me donne satisfaction pour 1,5 à 1,8 A en moyenne, mais je n’alimente rien d’autre que la traction en DCC.

      Il est cependant assez couteux par rapport à ses "compétiteur", de l’ordre de 30 €.

      J’aimerais avoir vos avis sur la solution Pololu.

      Répondre

      • La réponse à cela n’est pas simple ? Une première description de DCC++ par son concepteur semble être ici : www.trainboard.com. Ca commence par : « I’ve developed my own version of an open-source DCC controller. The base station consists of an Arduino Uno (or Arduino Mega) and an Arduino Motor Shield, and that’s the only hardware needed (with the exception of an 18V AC or similar power pack). » Il poursuit en parlant aussi d’une voie de programmation...

        Chargez DCC++ sur un UNO, rien que cela, raccordez y un shield L298, et vous avez une centrale, avec une sortie pour les voies. On ne peut pas plus simple, et qui fonctionne. C’est cet aspect qui m’a plu, en premier.

        Le Pololu, c’est avec deux ponts, donc un qui pourrait être utile pour une voie de programmation. C’est déjà plus compliqué tandis qu’avec les développements les plus récents (dont par l’équipe opendcc et nmra), il semble exister maintenant aussi la programmation directement sur voie, avec Railcom et un retour tel que via BiDib.

        Je suis loin d’avoir fini de tout étudier. Avec toutes ces briques, je compte moi aussi assembler un truc tout simple et fiable, pour oublier ensuite tous les aspects techniques. Probablement sans voie dédiée à la programmation. Ou sinon, avec un Arduino de plus, pour la programmation (occasionnelle). Donc un pont ou une sortie unique sur Arduino devrait me suffire.

        Répondre

        • « avec les développements les plus récents (dont par l’équipe opendcc et nmra), il semble exister maintenant aussi la programmation directement sur voie, avec Railcom et un retour tel que via BiDib. »

          Je suis très intéressé par ce que vous pourrez trouver sur cette question du RailCom, c’est à dire un retour du décodeur de loco dans une zone donnée, voire plus.

          Merci d’avance.

          Répondre

          • C’est l’étape d’après, avec des réalisations par ici, sur base PIC/Microchip : RailComDisplay
            Attention, car il faut des booters ou un signal DCC approprié, avec le "cut out", sinon, les décodeurs des locos vont fumer. C’est précisé sur la page.
            Par défaut, l’option Railcom est désactivée dans les décodeurs qui le supportent.

            Répondre

            • Merci pour ce lien : c’est très clair et je pense qu’on verra une version Arduino un de ces jours. Faire un cutout n’est pas plus difficile sur AVR que sur PIC.

              Si on ajoute une interface CAN...

              Répondre

              • j’ai toujours pensé que RailCom était à ce jour la version la plus aboutie pour la gestion de convois. Aujourd’hui, on peut savoir facilement que tel canton, telle section du réseau est occupée. Par ailleurs, avec un bon gestionnaire, DCC++ puisqu’on est dans son fil, il est aussi très facile de savoir l’état d’une loco (Arrêt, marche, sens, vitesse. Mais ce qui nous manque c’est les deux à la fois ! Je suis très intéressé, j’en ai déjà fait part sur le site et suis prêt à participer à tous travaux sur locoduino qui iraient dans le sens d’un développement sur Arduino et DCC++. Toutes mes locos sont par ailleurs compatibles RailCom !

                Répondre

  • Tout d’abord un grand merci à Dominique pour ce superbe article clair, précis, bien documenté (comme d’habitude) sur le logiciel DCC++. J’ai déjà eu recours à ses services lors d’essais (beaucoup plus laborieux) avec CmdrArduino.

    Personnellement, j’ai opté pour une solution avec deux UNO, le premier servant de centrale DCC++ (chargée avec le soft original, c’est à dire non modifié) et le deuxième utilisé comme commande à base d’une télécommande à infrarouge (télécommande d’un ancien lecteur de DVD). On a ainsi en une seule unité un clavier numérique, des flèches directionnelles et une série de boutons supplémentaires (gratuits).

    L’idée est de prévoir 4 boutons (1 par loco) sur la télécommande pour sélectionner facilement la machine souhaitée. La réalisation n’en est pas terminée mais j’aimerais partager une petite expérience vécue lors de la programmation de la carte qui héberge le programme "Centrale DCC++". Comme prévu, la pin RX de cette carte est reliée à la TX de la carte "Télecommande IR" pour transmettre les ordres via la liaison série ; or, lors du chargement du soft sur la carte "Centrale", j’obtiens en fin de compilation le message d’erreur suivant : " stk500_getsync() attempt 10 of 10 : not in sync : resp=0x41".
    Après recherches sur le net, il semblerait qu’il faille débrancher la connexion RX/TX pendant le chargement du soft. J’ai essayé et je n’ai plus l’erreur ! Je n’ai pas tout compris, mais cela fonctionne ! Si cela peut servir à d’autres...

    Daniel.

    Répondre

  • Bravo Daniel ;)

    Le problème du televersement est tout simple à comprendre :

    Le televersement se fait sur le port Tx-Rx (pins 0 et 1) de l’Arduino. Ce port est donc relié à l’interface USB pendant le televersement. Si vous le laissez relié à un autre montage pendant le televersement, il y a un conflit sur la pin Rx et la pin Tx envoie des choses inattendues vers le montage raccordé.

    Il faut donc débrancher Tx et surtout Rx lors du televersement.

    Ou alors branchez votre montage sur un autre port Série, avec la bibliothèque SoftwareSerial

    Répondre

  • Qu’en est-il de la rétrosignalisation 6 février 2017 11:38, par Pierre

    Bonjour à tous,
    Tout d’abord merci dominique pour ces explications claires et détaillées.
    Je viens vous voir aujourd’hui car mon père étant passionné de trains il nous met à contribution mon meilleur ami et moi pour la partie électronique (oui je sais ca fait beaucoup :p).
    Je me suis lancé pour défi de lui prouver que toutes les solutions techniques qui se trouvaient dans le commerce pouvaient être remplacées par une arduino. Je suis donc très heureux d’être tombé sur votre site.
    A la lecture de cet article tout ce dont j’ai besoin semble indiqué mais il me reste une question : je ne comprend pas bien comment recevoir des informations venant d’une locomotive (comme le ferai le railcom par exemple) est-ce par détection fine de courant ? Je n’ai pas encore trouvé de sujet parlant du "retour" d’informations des locomotives et je ne comprend pas trop le principe et la prioriété dans l’interruption...
    Pouvez-vous m’éclairer ?
    Merci beaucoup et bonne journée
    Amicalement
    Pierre

    Répondre

    • Qu’en est-il de la rétrosignalisation 6 février 2017 11:58, par Dominique

      Bonjour Pierre et merci pour vos encouragements.
      Votre question sur le "retour" d’information" des décodeurs est traitée en grande partie dans le forum, ici.
      La plupart des décodeurs savent répondre à une demande de lecture de CV, sur la voie de programmation évidemment (si plusieurs machines répondent en même temps, ce serait inexploitable) en provoquant une consommation de courant ou pas pour chaque bit de la réponse.
      C’est pour cela qu’on voit "gigoter" la loco. Parfois le décodeur arrive a répondre sans l’aide du moteur.
      Equipé d’un circuit de mesure de courant, DCC++ sait récupérer cette information.
      Personnellement je m’en sers pour détecter automatiquement l’adresse DCC (courte ou longue) d’une loco.
      Je ne suis pas loin de Railcom, mais c’est une fonction standard de DCC++.
      Nous avons constaté aussi que certains décodeurs ne répondent pas à cette demande qui, pourtant est dans la norme NMRA. Nous cherchons un explication.
      Evidemment on va chercher à intégrer Railcom aussi dans DCC++ ou dans un montage spécifique mais nous n’avons pas d’objectif daté pour le moment.

      Voir en ligne : http://forum.locoduino.org/index.ph...

      Répondre

      • Qu’en est-il de la rétrosignalisation 6 février 2017 12:14, par Pierre

        Merci Dominique pour cette réponse super rapide.
        Je suis très heureux d’être tombé sur vos projets qui aident à une compréhension pas à pas.
        Mon unique intérêt pour le railcom, c’est le fait de pouvoir détecter des locos perdues sur le réseau après une panne de courant par exemple. Cela permettrait de remettre en marche des trajets automatiques sans grande difficultés.
        Etant actuellement en prépa je n’ai malheureusement pas le temps d’étudier en profondeur tout ces gadgets pourtant primordiaux pour une utilisation fluide au quotidien ... Vivement cet été !
        Merci encore pour vos travaux et à bientôt !
        Pierre

        Répondre

      • Qu’en est-il de la rétrosignalisation 6 février 2017 12:31, par Dominique

        Pierre,

        Nous sommes tous dans le même cas, nous avons besoin de localiser les trains pour assurer la gestion du réseau.

        Il y a plusieurs techniques qui peuvent contribuer à cette détection :

        • Railcom bien-sur
        • la détection RFID
        • l’intelligence du gestionnaire
        • la combinaison des événements d’occupation/libération et de traction.

        Le sujet est cité à plusieurs endroits sur le Forum.
        Suivez-nous bien car on finira par y arriver :)

        Voir en ligne : http://forum.locoduino.org/index.ph...

        Répondre

        • Qu’en est-il de la rétrosignalisation 6 février 2017 13:15, par Pierre

          Comptez sur moi ;)

          Répondre

          • Qu’en est-il de la rétrosignalisation 4 mars 2017 16:41, par xavier

            Bonjour,
            Tout d’abord, un énorme Merci pour avoir porté à la connaissance de tous cet excellent projet.
            A mon tour, je me suis lancé, utilisant un Mega 260 et un shield moteur à base de L298. Je l’ai complété d’une rétro signalisation S88, que je tiens à la disposition de ceux qui pourraient être intéressés : elle utilise deux commandes, l’une pour indiquer le nombre de modules de rétro signalisation, il faut scanner, la seconde remontant l’état de tous les cantons quand un changement d’occupation est détecté.

            Utilisant CDM-Rail pour piloter mon réseau, j’espère que Jean-Pierre Pilou, l’auteur de ce logiciel, mettra prochainement un driver compatible Dcc++ à disposition (il est en train de monter sa commande Dcc++)

            J’ai aussi écrit un programme de commande en C#, lui aussi disponible à ceux qui le souhaitent.

            Amicalement,

            Xavier

            Voir en ligne : Ma page Dcc++

            Répondre

  • Booster multiples 4 mars 2017 16:48, par xavier

    Bonjour,
    Pour augmenter la puissance de Dcc++, j’envisage d’utiliser de multiples Shield à base de L298.
    Avec un shield, on disposerait de deux booster complémentaires de 2A chacun.
    Ils recevraient tous les commandes envoyées habituellement à la voie principale.
    La détection de court circuit se ferait pour chacun.
    Quelqu’un a-t-il déjà exploré cette voie ? Quelles contre indication voyez-vous ?

    Amicalement,
    Xavier

    Voir en ligne : Ma page Dcc++

    Répondre

  • Bonjour,

    Votre question est très pertinente surtout pour l’échelle HO où les machines consomment plus qu’en N.

    Je ne connais pas d’expérience, sur Locoduino, de mise en parallèle de ponts en H de type L298, mais ce doit être possible, en appliquant les même signaux de commande aux différents ponts. Mais il faut toujours se méfier des dispersions de caractéristiques des composants de puissance qui peuvent laisser des micro-court-circuits si les commutations ne sont pas strictement simultanées.

    Une autre solution consiste à utiliser des ponts en H plus puissants comme le LMD18200 qui peut facilement tenir 3A (6A en pointe).

    En cherchant des ampérages plus importants, vous pouvez vous orienter aussi vers ce produit qui tient 43A :
    BTS7960B-DC-43A
    J’en ai acheté un mais je ne l’ai pas encore testé.

    Bien-entendu, vous devrez soigner la détection de court-circuit, cela va de soi :)

    Cordialement
    Dominique

    Répondre

  • Bonjour
    Peut-on réaliser à partir de Arduino une interface entre CDM Rail et un booster Cobalt Box ?
    Cordialement
    Jocelyn

    Répondre

  • Bonjour,

    Je ne connais pas les interfaces de ces 2 produits. Si vous les décrivez je pourrais peut-être répondre.

    Cordialement

    Répondre

    • Bonjour
      le logiciel CDM rail est un logiciel gratuit permettant de tracer un réseau et de simuler la circulation de trains.
      Le booster Cobalt est celui vendu avec les moteurs d’aiguillage Cobalt avec interface IP. Il possède une entrée track power, une connection Main DCC system,
      une entrée DCC system control / Cab bus
      une connection pour leur système Alpha (commande des accessoires)
      une entrée pour l’alimentation
      une sortie Alpha bus / Cab bus extension
      une sortie Track Power de 5 ampères
      Merci
      Jocelyn

      Répondre

      • Jocelyn,

        Un petit schéma serait bien utile pour expliquer ce que vous voudriez faire avec un Arduino.
        Dans votre réponse, il n’y a pas de place pour un Arduino.
        Indiquez bien quelles fonctions vous voulez-lui faire faire, quelle interface utilisateur voulez-vous, sur quoi doit il agir, ...

        Là je ne peux pas deviner !

        Il vaudrait mieux continuer la discussion sur le Forum car nous sommes totalement hors sujet (ici c’est DCC++) et je vais rapidement effacer cette question.
        Merci de votre comprehension.

        Dominique

        Répondre

  • Etant en retraite, je dispose d’un peu de temps pour redémarrer la réalisation d’un réseau.
    Pour des raisons économiques, j’ai commencé par la création du plan sur CDM Rail, qui permet théoriquement de commander les trains et les accessoires.
    Je pense éviter la réalisation d’un TCO en utilisant ce logiciel.
    Je dispose d’un booster et d’une alimentation Cobalt qui m’a été livrée avec les moteurs d’aiguillages Cobalt IP.
    Ce booster dispose d’une entrée Bus Cab.
    Est il possible de réaliser une interface sur la base d’un Arduino entre ces deux systèmes me permettant dans un premier temps la commande des aiguillages ?
    Merci de votre réponse

    Répondre

    • Bon : j’insiste : allez sur le Forum et exposez votre question à nouveau depuis le début : c’est obligatoire pour ne pas polluer cet article sur DCC++ car je ne vois pas le rapport et vous non plus.
      J’efface votre question demain (DEMAIN) sur ce sujet mais je pense que votre question a de l’interêt donc allez au bon endroit SVP.
      Ce site est sympa et doit le rester.

      Dominique

      Répondre

  • Bonjour,

    Je recherche juste un exemple de programme prenant en compte la rétro-signalisation (une intégration des paramètres décrits par le concepteur) dans l’onglet "Sensor" de ce très bon programme fonctionnant à merveille pour l’ensemble locomotives, accessoires et signaux .

    Merci pour vos aides

    Cordialement

    Répondre

  • Bornes Arduino 18 juillet 2020 16:01, par Lionel

    Bonjour à tous,

    D’abord, un grand merci pour ce super projet. En m’appuyant sur la bibliothèque DCCpp j’ai réalisé une carte "mère" qui accueille un Arduino Mega 2560 ainsi que de l’électronique annexe pour des sorties, et également une sortie pour le DCC avec le schéma pour un LMD 18200. En revanche, sur mon PCB, les sorties routées sur le LMD 18200 ne sont pas celles citées en exemple dans le tableau, mais sur les broches 2 et 13, au lieu des 3 et 12.

    Dans mon programme, j’appelle bien DCCpp::beginMain(UNDEFINED_PIN, 2, 13, UNDEFINED_PIN) (pour l’instant, je n’ai pas connecté le MAX471 le temps de mettre au point le système). Seulement, voilà, cela ne fonctionne pas, je n’ai pas de courant en sortie du LMD18200 pas plus que de mouvement de machine sur le réseau. En revanche, à titre d’essai, sans passer par mon PCB, en paramétrant les broches 3 et 12, cela fonctionne correctement.

    Sauriez-vous m’indiquer ce que j’ai loupé ?

    Merci et bonne journée.

    Lionel

    Répondre

  • Bonjour, je n’en sais pas assez pour vous donner les détails, mais des broches sont choisies en fonction du timer qui y est attaché.
    Impossible d’en changer pour des broches qui ne sont pas attachées à un timer.

    • Pins 5 and 6 : controlled by Timer 0
    • Pins 9 and 10 : controlled by timer 1
    • Pins 11 and 3 : controlled by timer 2
      Par ailleurs, bien que semblant libres, d’autres sont en fait utilisées par le programme.

    Répondre

  • Vous ne pouvez pas changer la broche 12 sur un Mega car elle correspond au timer 1 utilisé pour génèrer le signal DCC :

    Arduino MEGA :

    Timer 0 : pins 4 et 13
    Timer 1 : pins 11 et 12
    Timer 2 : pins 9 et 10

    La deuxième broche du timer doit rester libre : on ne peut pas s’en servir pour des E/S (il en reste assez sur un Mega.

    Répondre

  • Comment lire les capteurs ? 16 juillet 2021 15:57, par jean-jacques Goessens

    Bonjour,
    Novice en DCC, j’ai potassé et pense avoir compris le concept, nouveau pour moi.
    Toutefois, il me este une question :
    Comment sont gérés les capteurs d’un point de vue matériel ?
    je vois que l’on peut gérer les actionneurs à l’aide de modules DCC qui sont connectés sur les rails, mais je trouve peu de littérature pour les capteurs ?
    Quelle est la façon la plus "propre" de connecter mes capteurs, une autre carte Arduino sur un autre port USB, des modules déportés ? des modules DCC ?

    Répondre

    • Comment lire les capteurs ? 16 juillet 2021 23:17, par msport

      Les entrées des Arduino sont toutes faites pour recevoir des capteurs. Décrivez votre projet sur le forum, rubrique "Vos projets" pour avoir des réponses adaptées. (et regardez déjà ce qui est décrit sur le site)

      Répondre

  • Les réponses abondent sur le site : cherchez encore.

    Répondre

Réagissez à « Réalisation de centrales DCC avec le logiciel libre DCC++ (1) »

Qui êtes-vous ?
Votre message

Pour créer des paragraphes, laissez simplement des lignes vides.

Lien hypertexte

(Si votre message se réfère à un article publié sur le Web, ou à une page fournissant plus d’informations, vous pouvez indiquer ci-après le titre de la page et son adresse.)

Rubrique « Projets »

Les derniers articles

Les articles les plus lus