dimanche 1 décembre 2019

Analyse d'une communication SPI



Développé par Motorola dans les années 1980, SPI (acronyme de Serial Peripheral Interface) est un protocole de communication série conçu pour permettre à un microcontrôleur (le "maître") de communiquer à haute vitesse avec un ou plusieurs périphériques (les "esclaves").

Une communication SPI implique 4 lignes de transmissions: MOSI, MISO, SCLK et CS.

1) MOSI (Master Out, Slave In) est utilisée pour la transmission de données du microcontrôleur vers le périphérique. Elle est contrôlée par le microcontrôleur maître.

2) MISO (Master In, Slave Out) est utilisée pour la transmission de données du périphérique vers le microcontrôleur. C'est la seule des quatre lignes qui est contrôlée par le périphérique esclave.

3) SCLK (Serial Clock) transmet un signal d'horloge généré par le microcontrôleur maître. Il sert à synchroniser  l'esclave avec le maître. Contrairement à la communication UART, la communication SPI est donc une communication synchrone.

4) CS ou SS (Chip Select ou Slave Select) est utilisée pour activer un esclave. Contrairement à MOSI, MISO et SCLK, qui sont partagées par tous les périphériques esclaves branchés au microcontrôleur maître, chaque périphérique esclave doit avoir sa propre ligne CS. La ligne CS d'un périphérique doit être mise au niveau logique BAS pour que ce périphérique tienne compte des messages envoyés par le maître sur la ligne MOSI. De cette façon, le maître communique avec un esclave à la fois.

C'est une communication full duplex, ce qui signifie que des données circulent du maître à l'esclave et de l'esclave au maître au même moment.

Sur un Arduino Uno, MOSI est la broche 11, MISO est la broche 12 et SCLK est la broche 13. On utilise souvent la broche 10 pour CS, mais ce n'est évidemment pas obligatoire puisque plusieurs lignes CS différentes seront nécessaire en présence de plusieurs périphériques esclaves.

Des exemples de périphériques qui communiquent en SPI

Voici quelques exemples de périphériques qui communiquent en SPI avec un Arduino: lecteur de cartes SD, module de communicaton radio nRF24L01module RFID-RC522lecteur de fichiers mp3 VS1053, écran couleur ST7735, afficheur TM1638, potentiomètre numérique MCP41100, etc.

Observation d'une transaction SPI

Pour observer une communication SPI au moyen d'un analyseur logique, j'ai branché un convertisseur analogique-numérique (ADC) MCP3008 à un Arduino Uno. Ce circuit intégré comporte 8 entrées analogiques dont l'état est transmis de façon numérique par une communication SPI.
J'ai branché le MCP3008 à l'Arduino de la façon suivante:

  • Broche 5 du MCP3008: potentiomètre permettant de faire varier la tension entre 0 et  5 V.
  • Broches 9 et 14 du MCP3008: GND de l'Arduino
  • Broches 15 et 16 du MCP3008: 5 V de l'Arduino
  • Broche 10 du MCP3008: broche 10 de l'Arduino et canal 0 de l'analyseur logique
  • Broche 11 du MCP3008: broche 11 de l'Arduino et canal 3 de l'analyseur logique
  • Broche 12 du MCP3008: broche 12 de l'Arduino et canal 2 de l'analyseur logique
  • Broche 13 du MCP3008: broche 13 de l'Arduino et canal 1 de l'analyseur logique


L'ajout d'un analyseur logique m'a permis de visualiser la communication SPI avec le logiciel Pulseview.

Le sketch que j'ai utilisé est présenté plus loin dans cet article: il consiste à établir une connexion avec le MCP3008 et de lui demander la valeur mesurée sur son canal 4.

Puisque le MCP3008 est un ADC à 10 bits, les valeurs obtenues peuvent varier entre 0 et 1023. Le potentiomètre a d'abord été réglé de façon à produire une valeur de 562:


Voici une transaction SPI au cours de laquelle l'Arduino demande la valeur du canal 4, ainsi que la réponse du MCP3008 (vous pouvez agrandir l'image en cliquant dessus). Remarquez que la ligne CS, qui était initialement l'état logique HAUT, se met temporairement à l'état logique BAS pendant toute la durée de la transaction.

Dans le cas du MPC3008, cette transaction nécessite l'échange de 3 octets. L'horloge (SCLK) a donc accompli trois séries consécutives de 8 oscillations. L'état logique des lignes MOSI et MISO est lu à chaque front ascendant de l'horloge.



Analysons chacun des 3 octets d'une façon plus détaillée:

Le premier octet est le message d'initialisation envoyé par le microcontrôleur au périphérique (sur la ligne MOSI, donc): il s'agit du nombre binaire "00000001". Pendant cette phase, l'état de la ligne MISO n'a aucune importance. Cet octet est émis à la ligne 41 du sketch (voir plus bas). Sur le diagramme ci-dessous, j'ai ajouté des gros points noirs pour indiquer à quel moment ces valeurs sont lues, pendant le front ascendant du signal d'horloge:

Dans le deuxième octet, le microcontrôleur indique au périphérique l'adresse de l'entrée analogique dont il désire lire la valeur. Selon la fiche technique du MCP3008, pour lire le canal numéro 4, il faut que les 4 premiers bits sur la ligne MOSI soient "1100"; la valeur des 4 derniers bits n'a aucune importance (on envoie généralement 0); ce signal est généré à la ligne 49 du sketch :


Sur la ligne MISO, le périphérique utilise les deux derniers bits de ce deuxième octet pour transmettre les deux premiers bits de sa réponse (qui en comportera ultimement 10). Ici, ces deux bits sont 1 et 0:

Finalement, les 8 derniers bits de la réponse du périphérique sont communiqués dans le troisième octet, sur la ligne MISO. Pour ce troisième octet, l'état de la ligne MOSI n'a aucune importance. Ici, le signal reçu est 00110010.


En mettant bout à bout les deux derniers bits du deuxième octet (10) et les 8 bits du troisième octet (00110010), nous obtenons le nombre binaire 1000110010, ou 562 en décimal: c'est bien à cette valeur que le potentiomètre avait été réglé.

Voici une transaction lorsque le potentiomètre est à la position minimale (0); les dix derniers bits retournés sur la ligne MISO sont 0000000000:


...et une autre lorsque le potentiomètre est à la position maximale (1023); les dix derniers bits retournés sur la ligne MISO sont 1111111111:


Le sketch utilisé

Sur Arduino, la communication SPI est gérée par la bibliothèque du même nom, qui est fournie par défaut avec l'IDE Arduino.

À la ligne 35 du sketch ci-dessous, on initie une transaction SPI en spécifiant 3 paramètres:la fréquence, le boutisme et le mode SPI (dans ce cas: 2 MHz, le bit de poids fort en premier, et le mode SPI 0):

SPI.beginTransaction (SPISettings(2000000, MSBFIRST, SPI_MODE0));

Un Arduino Uno peut sans problème établir une communication SPI à une fréquence de 4 MHz, mais il faut également tenir compte de la fréquence maximale que peut supporter le périphérique. Des fils conducteurs trop longs peuvent aussi nous obliger à diminuer la fréquence. Il existe 4 modes de communication SPI. Le plus fréquemment utilisé est le mode 0, dans lequel l'horloge est au niveau logique BAS quand elle est inactive, et la lecture des données s'effectue pendant le front montant de l'horloge.

SPI.transfer() permet simultanément l'envoi d'un octet du maître vers l'esclave sur la ligne MOSI et la réception d'un octet de l'esclave vers le maître sur la ligne MISO:

octet_recu = SPI.transfer(octet_envoyé);

À la ligne 41, on envoie le premier des 3 octets (00000001). Puisque la réponse de l'ADC est sans importance à cette étape, on ne se donne pas la peine de la stocker dans une variable.

SPI.transfer (0b00000001);

Le deuxième octet est envoyé à la ligne 45. C'est la commande qui indique qu'on désire lire le canal numéro 4. Cette fois, la réponse est stockée dans la variable octet_recu_1, puisque les deux derniers bits de la réponse constituent une information utile.

octet_recu_1 = SPI.transfer(0b11000000);

À la ligne 49, on envoie un troisième octet qui n'est qu'une suite de zéros, puisque ce message sera ignoré par le MCP3008. C'est strictement le message reçu qui nous intéresse: nous le stockons dans la variable octet_recu_2:

octet_recu_2 = SPI.transfer(0b00000000);

La ligne 59 consiste à mettre, dans une même variable, les deux derniers bits de la variable octet_recu_1, suivis des 8 bits de la variable octet_recu_2. Si la syntaxe vous donne du fil à retordre, relisez l'article sur les opérations bit à bit!

resultat = (octet_recu_1 & 0b00000011)<< 8 | octet_recu_2;

-
-

À lire aussi

Deux autres types de communication série ont également été abordés analyse d'une communication UART et analyse d'une communication I2C.

Le MCP3008 n'en est pas à sa première apparition dans ce blog: je l'avais utilisé afin d'ajouter des entrées analogiques au Raspberry Pi ainsi qu'à l'ESP8266.

Yves Pelletier   (TwitterFacebook)


Aucun commentaire:

Enregistrer un commentaire