

# MINF Programmation des PIC32MX

# **Chapitre 8**

Gestion du bus SPI



Christian HUBER (CHR) Serge CASTOLDI (SCA) Version 1.91 mars 2019



# CONTENU DU CHAPITRE 8

| 8. <i>Ges</i> | stion du bus SPI avec le PIC32MX                                                                 | 8-1          |
|---------------|--------------------------------------------------------------------------------------------------|--------------|
| 8.1.          | Bus SPI                                                                                          | 8-2          |
| 8.1.1.        |                                                                                                  |              |
| 8.1.2.        |                                                                                                  |              |
| 8.1.3.        | Connexions multi-master                                                                          | 8-3          |
| 8.1.4.        |                                                                                                  |              |
| 8.2.          | Réalisation du bus SPI avec le PIC32MX                                                           | 8-4          |
| 8.2.1.        |                                                                                                  | 8-4          |
| 8.2.2.        |                                                                                                  | 8-5          |
| 8.2.3.        |                                                                                                  |              |
| 8.2.4.        |                                                                                                  |              |
| 8.2.5.        | Timing en mode master (8 bits)                                                                   | 8-7          |
| 8.2.6.        |                                                                                                  |              |
| 8.2           | .6.1. Slave Select pin disabled                                                                  |              |
| 8.2           | .6.2. Slave Select pin enabled                                                                   | 8-9          |
| 8.3.          | Les fonctions SPI de la PLIB_SPI                                                                 | 8-10         |
| 8.3.1.        | C                                                                                                | 8-10         |
|               | .1.1. La fonction PLIB_SPI_BaudRateClockSelect                                                   |              |
|               | .1.2. La fonction PLIB_SPI_BaudRateSet                                                           | 8-10         |
|               | .1.3. La fonction PLIB_SPI_CommunicationWidthSelect                                              | 8-11         |
|               | .1.4. La fonction PLIB_SPI_ClockPolaritySelect                                                   | 8-11         |
|               | .1.5. La fonction PLIB_SPI_InputSamplePhaseSelect1.6. La fonction PLIB_SPI_OutputDataPhaseSelect | 8-11<br>9 12 |
|               | 1                                                                                                |              |
|               | .1.7. La fonction PLIB_SPI_PinEnable                                                             |              |
|               | 1.9. La fonction PLIB_SPI_FIFOInterruptModeSelect                                                |              |
|               | Les fonctions du transmetteur                                                                    |              |
|               | Les fonctions du récepteur                                                                       |              |
|               | Les fonctions de "Data Transfer"                                                                 |              |
|               | .4.1. La fonction PLIB_SPI_BufferClear                                                           |              |
|               | .4.2. La fonction PLIB_SPI_BufferRead                                                            |              |
|               | .4.3. La fonction PLIB_SPI_BufferWrite                                                           |              |
| 8.4.          | Etude du driver SPI fourni par MHC                                                               | 8-15         |
| 8.4.1.        |                                                                                                  |              |
| 8.4.2.        |                                                                                                  |              |
| 8.4.3.        |                                                                                                  | 8-16         |
| 8.4.4.        |                                                                                                  |              |
| 8.4.5.        |                                                                                                  |              |
| 8.5.          | Réalisation de l'utilitaire SPI                                                                  | 8-18         |
| 8.5.1.        |                                                                                                  |              |
| 8.5.2.        |                                                                                                  |              |
|               | .2.1. Séquence des actions de lecture                                                            |              |
|               | .2.2. Observation du moment de lecture du tampon                                                 | 8-20         |



| <b>8.7.</b> ] | Périphériques SPI du Kit PIC32MX                           | 8-21 |
|---------------|------------------------------------------------------------|------|
| 8.8.          | Communication avec le Dac LTC2604                          | 8-22 |
| 8.8.1.        | Chip Select du LTC2604                                     | 8-22 |
| 8.8.2.        | Configuration SPI nécessaire au LTC2604                    | 8-22 |
| 8.8.2         |                                                            | 8-23 |
| 8.8.3.        | Détail du SPIxCON                                          | 8-25 |
| 8.8.4.        | Sélection de la fréquence de SCK                           | 8-27 |
| 8.8.5.        | Initialisation du LTC2604                                  | 8-27 |
| 8.8.6.        | Ecriture d'une valeur sur le LTC2604                       | 8-28 |
| 8.8.7.        |                                                            |      |
| 8.8.7         |                                                            |      |
| 8.8.7         | .2. Signal sur le DAC                                      | 8-30 |
| 8.10.         | Communication avec le LM70                                 | 8-31 |
| 8.10.1.       |                                                            | 8-31 |
| 8.10.2.       | Configuration SPI nécessaire au LM70                       | 8-31 |
|               | 2.1. Fonction de configuration du SPI pour le LM70         | 8-32 |
| 8.10.         | 2.2. Vérification de la configuration avec SPI1CON         |      |
| 8.10.3.       |                                                            |      |
| 8.10.4.       | Lecture du LM70                                            | 8-34 |
| 8.10.5.       | La fonction SPI_ReadRawTempLM70                            | 8-34 |
| 8.10.6.       | Ecriture vers le LM70                                      | 8-35 |
| 8.10.         | 6.1. Exemple d'écriture                                    |      |
| 8.10.7.       |                                                            |      |
| 8.10.         | 7.1. Vue de détail                                         | 8-37 |
| <b>8.11.</b>  | Utilisation combinée des 2 slaves SPI                      | 8-38 |
| 8.11.1.       | Fonction lecture LM70 avec reconfiguration                 | 8-38 |
| 8.11.2.       |                                                            | 8-39 |
| 8.11.3.       |                                                            | 8-39 |
| 8.11.         | 3.1. Contenu de la réponse à l'interruption du Timer1      | 8-39 |
| 8.11.         | 3.2. Affichage température dans l'application              |      |
| 8.11.         | 3.3. Problème avec les variables float dans l'interruption |      |
| 8.11.4.       | Remarque sur les reconfigurations                          | 8-41 |
| <b>8.12.</b>  | Fichiers à disposition                                     | 8-41 |
| 8.13.         | Conclusion                                                 | 8-41 |
|               | Historique des versions                                    |      |
| 8.14.1.       |                                                            |      |
| 8.14.2.       |                                                            |      |
| 8.14.3.       |                                                            |      |
| 8.14.4.       |                                                            |      |
| 8.14.5.       |                                                            |      |
| 8.14.6.       |                                                            |      |
| 8.14.7.       |                                                            |      |
| 0.14./.       | v Ci 510H 1.71 Hidi 5 2017                                 | 0-42 |



# 8. GESTION DU BUS SPI AVEC LE PIC32MX

Ce chapitre traite du bus SPI et de sa gestion avec le microcontrôleur PIC32MX795F512L ainsi que de la mise en pratique avec les composants LM70 et LTC2604.

Le PIC32MX dispose de modules de communication sérielle synchrone dédiés à la communication SPI (Serial Peripheral Interface).

Les documents de référence sont :

- La documentation "PIC32 Family Reference Manual" : Section 23 : Serial Peripheral Interface
- Pour les détails spécifiques au PIC32MX795F512L, il faut se référer au document "PIC32MX5XX/6XX/7XX Family Data Sheet":
   Section 18: Serial Peripheral Interface (SPI)
- La documentation d'Harmony, qui se trouve dans <Répertoire Harmony>\v<n>\doc: Section MPLAB Harmony Framework Reference > Peripheral Libraries Help, sous-section SPI Peripheral Library



Ce document a été établi sur la base de Harmony v1.06.



# **8.1. BUS SPI**

Avant d'étudier comment gérer le bus SPI avec les microcontrôleurs PIC, il est nécessaire de connaître le bus SPI.

Nous allons présenter le rôle des lignes du bus, les modes et les signaux.

# 8.1.1. LES LIGNES DU BUS SPI

Voici les connexions standards sur un composant SPI, donc esclave.



# **8.1.2.** Connexions entre master et slaves



Motorola définit les lignes de la manière suivante :

- MISO = Master In, Slave Out.
- MOSI = Master Out, Slave In
- SCK = Serial ClocK
- SS = Slave Select



# **8.1.3.** CONNEXIONS MULTI-MASTER

Il ne doit y avoir qu'un seul maître à la fois, mais les rôles peuvent être permutés. Cela implique le passage pour le SCK de sortie à entrée lorsque le maître devient esclave.



# **8.1.4.** LES MODES MASTER ET SLAVE

En mode maître, le composant (en général le microcontrôleur) doit fournir l'horloge et activer la sélection de l'esclave.

En mode esclave, le composant (en général un périphérique) reçoit l'horloge et réagit à l'entrée de sélection.

En général un microcontrôleur peut être configuré en maître ou en esclave, c'est le cas du PIC.



# 8.2. RÉALISATION DU BUS SPI AVEC LE PIC32MX

Le PIC32MX795F512L possède 4 modules SPI. Voici l'attribution des broches.

|          |                    | in Number <sup>(</sup> | 1)              | Pin  | Buffer |                                                |
|----------|--------------------|------------------------|-----------------|------|--------|------------------------------------------------|
| Pin Name | 64-Pin<br>QFN/TQFP | 100-Pin<br>TQFP        | 121-Pin<br>XBGA | Туре | Туре   | Description                                    |
| SCK1     | _                  | 70                     | D11             | I/O  | ST     | Synchronous serial clock input/output for SPI1 |
| SDI1     | _                  | 9                      | E1              | I    | ST     | SPI1 data in                                   |
| SDO1     | _                  | 72                     | D9              | 0    | _      | SPI1 data out                                  |
| SS1      | _                  | 69                     | E10             | I/O  | ST     | SPI1 slave synchronization or frame pulse I/O  |
| SCK3     | 49                 | 48                     | K9              | I/O  | ST     | Synchronous serial clock input/output for SPI3 |
| SDI3     | 50                 | 52                     | K11             | I    | ST     | SPI3 data in                                   |
| SDO3     | 51                 | 53                     | J10             | 0    | _      | SPI3 data out                                  |
| SS3      | 43                 | 47                     | L9              | I/O  | ST     | SPI3 slave synchronization or frame pulse I/O  |
| SCK2     | 4                  | 10                     | E3              | I/O  | ST     | Synchronous serial clock input/output for SPI2 |
| SDI2     | 5                  | 11                     | F4              | I    | ST     | SPI2 data in                                   |
| SDO2     | 6                  | 12                     | F2              | 0    | _      | SPI2 data out                                  |
| SS2      | 8                  | 14                     | F3              | I/O  | ST     | SPI2 slave synchronization or frame pulse I/O  |
| SCK4     | 29                 | 39                     | L6              | I/O  | ST     | Synchronous serial clock input/output for SPI4 |
| SDI4     | 31                 | 49                     | L10             | - 1  | ST     | SPI4 data in                                   |
| SDO4     | 32                 | 50                     | L11             | 0    |        | SPI4 data out                                  |
| SS4      | 21                 | 40                     | K6              | I/O  | ST     | SPI4 slave synchronization or frame pulse I/O  |

# 8.2.1. CÂBLAGE DU SPI SUR LE KITPIC32MX

Comme on peut le voir dans les éléments de schéma ci-dessous, le kit utilise le module SPI 1 :



| Broche du PIC32MX795F512L<br>(100 pin TQFP) | Nom schéma | No broche<br>boitier 100 |
|---------------------------------------------|------------|--------------------------|
| SCK1/RD10                                   | SPI-SCK    | 70                       |
| SDO1/RD0                                    | SPI-MOSI   | 72                       |
| SDI1/RC4                                    | SPI-MISO   | 9                        |
| SS1/RD9                                     | DAC_CLR    | 69                       |

Remarque : Le slave select 1 \_SS1 est utilisé pour le DAC !



# 8.2.2. SCHÉMA-BLOC DU MODULE SPI

Voici le schéma bloc du module SPI:



Note: Access SPIxTXB and SPIxRXB FIFOs via SPIxBUF register.

# 8.2.3. PARTICULARITÉ DU SPIXSR

Comme le montre l'extrait ci-dessous, le registre à décalage SPIxSR va sortir un bit du registre pour chaque bit reçu. Cela signifie que lors de l'écriture, le slave fournit une info. Une transmission SPI est full-duplex.



Cette particularité est à gérer en relation avec les spécificités de chaque slave SPI.



# 8.2.4. LES POSSIBILITÉS DU MODULE SPI

Le tableau ci-dessous résume les possibilités du module SPI.

Table 23-1: SPI Features

| Available<br>SPI Modes | SPI<br>Master | SPI<br>Slave | Frame<br>Master | Frame<br>Slave | 8-Bit,<br>16-Bit,<br>and<br>32-Bit<br>Modes | Selectable<br>Clock<br>Pulses<br>and Edges | Selectable<br>Frame<br>Sync<br>Pulses<br>and Edges | Slave<br>Select<br>Pulse |
|------------------------|---------------|--------------|-----------------|----------------|---------------------------------------------|--------------------------------------------|----------------------------------------------------|--------------------------|
| Normal Mode            | Yes           | Yes          | _               | _              | Yes                                         | Yes                                        | _                                                  | Yes                      |
| Framed Mode            | Yes           | Yes          | Yes             | Yes            | Yes                                         | Yes                                        | Yes                                                | No                       |

On en déduit la possibilité de travailler en mode 8, 16 ou 32 bits. Le mode "Framed" ne sera pas traité dans ce chapitre.



# 8.2.5. TIMING EN MODE MASTER (8 BITS)

Figure 23-9: SPI Master Mode Operation in 8-bit Mode (MODE32 = 0, MODE16 = 0)



On peut observer les possibilités de configuration : les flancs de l'horloge en relation avec la donnée, ce qui permet d'adapter le master à un slave demandant un des modes. En mode 8 bits, le module SPI sérialise/dé-sérialise un octet à la fois.



### 8.2.6. TIMING EN MODE SLAVE

### 8.2.6.1. SLAVE SELECT PIN DISABLED

Figure 23-10: SPI Slave Mode Operation in 8-bit Mode with Slave Select Pin Disabled (MODE32 = 0, MODE16 = 0, SSEN = 0)



- (SPIxCON<8>). Any combination of CKP and CKE bits can be chosen for module operation.
  - 2: If there are no pending transmissions or a transmission is in progress, SPIxBUF is transferred to SPIxSR as soon as the user writes to SPIxBUF.
  - 3: Operation for 8-bit mode is shown. 16-bit and 32-bit modes are similar.



# 8.2.6.2. SLAVE SELECT PIN ENABLED

Figure 23-11: SPI Slave Mode Operation in 8-bit Mode with Slave Select Pin Enabled (MODE32 = 0, MODE16 = 0, SSEN = 1)





# 8.3. LES FONCTIONS SPI DE LA PLIB SPI

Avec l'introduction de Harmony, les fonctions des librairies périphériques ne font plus partie du compilateur. La PLIB\_SPI est décrite au paragraphe SPI Peripheral Library, dans la documentation de Harmony (version 1.06 ci-dessous).

### 8.3.1. LES FONCTIONS DE CONFIGURATION

Nous disposons de nombreuses fonctions de configuration dont la description sera limitée à celles demandant des explications.

| Name                              | Description                                                             |
|-----------------------------------|-------------------------------------------------------------------------|
| PLIB_SPI_BaudRateClockSelect      | Selects the type of clock is used by the Baud Rate Generator.           |
| PLIB_SPI_BaudRateSet              | Sets the baud rate to the desired value.                                |
| PLIB_SPI_ClockPolaritySelect      | Enables clock polarity.                                                 |
| PLIB_SPI_CommunicationWidthSelect | Selects the data width for the SPI communication.                       |
| PLIB_SPI_Disable                  | Disables the SPI module.                                                |
| PLIB_SPI_Enable                   | Enables the SPI module.                                                 |
| PLIB_SPI_ErrorInterruptDisable    | Enables SPI error interrupts.                                           |
| PLIB_SPI_ErrorInterruptEnable     | Enables SPI error interrupts                                            |
| PLIB_SPI_FIFOCountGet             | Reads the SPI Buffer Element Count bits for either receive or transmit. |
| PLIB_SPI_FIFODisable              | Disables the SPI enhanced buffer.                                       |
| PLIB_SPI_FIFOEnable               | Enables the SPI enhanced buffer.                                        |
| PLIB_SPI_FIFOInterruptModeSelect  | Selects the SPI buffer interrupt mode.                                  |
| PLIB_SPI_FIFOShiftRegisterIsEmpty | Returns the current status of the SPI shift register.                   |
| PLIB_SPI_InputSamplePhaseSelect   | Selects the SPI data input sample phase.                                |
| PLIB_SPI_IsBusy                   | Returns the current SPI module activity status.                         |
| PLIB_SPI_MasterEnable             | Enables the SPI in Master mode.                                         |
| PLIB_SPI_OutputDataPhaseSelect    | Selects serial output data change.                                      |
| PLIB_SPI_PinDisable               | Enables the selected SPI pins.                                          |
| PLIB_SPI_PinEnable                | Enables the selected SPI pins.                                          |
| PLIB_SPI_PrescalePrimarySelect    | Selects the primary prescale for SPI Master mode.                       |
| PLIB_SPI_PrescaleSecondarySelect  | Selects the secondary prescale for SPI Master mode.                     |
| PLIB_SPI_ReadDataIsSignExtended   | Returns the current status of the receive (RX) FIFO sign-extended data. |
| PLIB_SPI_SlaveEnable              | Enables the SPI in Slave mode.                                          |
| PLIB_SPI_SlaveSelectDisable       | Disables Master mode slave select.                                      |
| PLIB_SPI_SlaveSelectEnable        | Enables Master mode slave select.                                       |
| PLIB_SPI_StopInIdleDisable        | Continues module operation when the device enters Idle mode.            |
| PLIB_SPI_StopInIdleEnable         | Discontinues module operation when the device enters Idle mode.         |

### 8.3.1.1. LA FONCTION PLIB\_SPI\_BAUDRATECLOCKSELECT

© Cette fonction n'est pas supportée par le PIC32MX.

### LA FONCTION PLIB\_SPI\_BAUDRATESET 8.3.1.2.

La fonction PLIB\_SPI\_BaudRateSet, permet d'établir la fréquence du SCK. Voici son prototype:

```
void PLIB SPI BaudRateSet(SPI MODULE ID index,
                          uint32 t clockFrequency,
                          uint32 t baudRate)
```



Il faut fournir la fréquence de l'horloge (en principe PB\_CLOCK) ainsi que la fréquence voulue. Il faut cependant veiller à ce que le rapport des fréquences soit un nombre entier pair (2, 4, 6, 8, ...).

Par exemple avec l'horloge à 80 MHz pour obtenir SCK à 20 MHz le rapport est de 4, on écrira :

```
PLIB SPI BaudRateSet(KitSpi1, SYS CLK FREQ, 20000000);
```

# 8.3.1.3. LA FONCTION PLIB\_SPI\_COMMUNICATION WIDTH SELECT

La fonction PLIB\_SPI\_CommunicationWidthSelect permet d'établir si la communication est effectuée par paquets de 8, 16 ou 32 bits. Voici son prototype :

Le type énuméré SPI\_COMMUNICATION\_WIDTH permet le choix.

```
typedef enum {
    SPI_COMMUNICATION_WIDTH_8BITS = 0,
    SPI_COMMUNICATION_WIDTH_16BITS = 1,
    SPI_COMMUNICATION_WIDTH_32BITS = 2
} SPI_COMMUNICATION_WIDTH;
```

# 8.3.1.4. LA FONCTION PLIB\_SPI\_CLOCKPOLARITYSELECT

La fonction **PLIB\_SPI\_ClockPolaritySelect** permet de sélectionner la polarité de l'horloge.

Le type énuméré SPI\_CLOCK\_POLARITY permet le choix entre 2 polarités :

```
typedef enum {
     SPI_CLOCK_POLARITY_IDLE_LOW = 0,
     SPI_CLOCK_POLARITY_IDLE_HIGH = 1
} SPI_CLOCK_POLARITY;
```

Cette fonction agit sur le bit CKP du registre SPIxCON.

```
bit 6 CKP: Clock Polarity Select bit

1 = Idle state for clock is a high level; active state is a low level

0 = Idle state for clock is a low level; active state is a high level
```

# 8.3.1.5. LA FONCTION PLIB SPI INPUTSAMPLEPHASESELECT

La fonction **PLIB\_SPI\_InputSamplePhaseSelect** permet d'établir à quel moment les bits en entrée sont échantillonnés. Son prototype est le suivant :

Le type énuméré SPI\_INPUT\_SAMPLING\_PHASE permet le choix entre 2 variantes :

```
typedef enum {
    SPI_INPUT_SAMPLING_PHASE_IN_MIDDLE = 0,
    SPI_INPUT_SAMPLING_PHASE_AT_END = 1
} SPI_INPUT_SAMPLING_PHASE;
```

Cette fonction agit sur le bit SMP du registre SPIxCON.



```
bit 9

SMP: SPI Data Input Sample Phase bit

Master mode (MSTEN = 1):

1 = Input data sampled at end of data output time
0 = Input data sampled at middle of data output time
Slave mode (MSTEN = 0):

SMP value is ignored when SPI is used in Slave mode. The module always uses SMP = 0.
```

# 8.3.1.6. LA FONCTION PLIB\_SPI\_OUTPUTDATAPHASESELECT

La fonction **PLIB\_SPI\_OutputDataPhaseSelect** permet d'établir sur quelle transition de l'horloge les données sont émises. Son prototype est le suivant :

Le type énuméré SPI\_OUTPUT\_DATA\_PHASE permet d'indiquer le choix.

```
typedef enum {
    SPI_OUTPUT_DATA_PHASE_ON_IDLE_TO_ACTIVE_CLOCK = 0,
    SPI_OUTPUT_DATA_PHASE_ON_ACTIVE_TO_IDLE_CLOCK = 1
} SPI_OUTPUT_DATA_PHASE;
```

Cette fonction agit sur le bit CKE du registre SPIxCON.

```
bit 8 CKE: SPI Clock Edge Select bit

1 = Serial output data changes on transition from active clock state to Idle clock state (see CKP bit)

0 = Serial output data changes on transition from Idle clock state to active clock state (see CKP bit)
```

# 8.3.1.7. LA FONCTION PLIB\_SPI\_PINENABLE

La fonction PLIB\_SPI\_PinEnable permet d'activer les lignes SPI mentionnées.

```
void PLIB_SPI_PinEnable(SPI_MODULE_ID index, SPI_PIN pin)
```

Le type énuméré SPI\_PIN permet d'indiquer la ligne à activer.

```
typedef enum {
   SPI_PIN_DATA_OUT,
   SPI_PIN_DATA_IN,
   SPI_PIN_SLAVE_SELECT
} SPI_PIN;
```

# 8.3.1.8. LA FONCTION PLIB\_SPI\_ISBUSY

La fonction **PLIB\_SPI\_IsBusy** n'est pas une fonction de configuration, elle permet de savoir si le system SPI est actif (sérialisation/desérialisation ou au repos).

```
bool PLIB_SPI_IsBusy(SPI_MODULE_ID index);
```

# Returns

- true SPI module is currently busy with some transactions
- · false SPI module is currently idle



# 8.3.1.9. LA FONCTION PLIB\_SPI\_FIFOINTERRUPTMODESELECT

La fonction **PLIB\_SPI\_FIFOInterruptModeSelect** permet de configurer le comportement de l'interruption SPI en relation avec la situation des tampons de réception et d'émission, ceci pour autant que l'on ait activé l'utilisation du buffer avancé (à l'aide de la fonction **PLIB\_SPI\_FIFOEnable**).

Son prototype est le suivant :

Le type énuméré SPI\_FIFO\_INTERRUPT permet les choix suivants :

L'utilisation de l'interruption est particulièrement utile en mode SLAVE pour réagir rapidement aux données reçues afin de répondre.

# **8.3.2.** LES FONCTIONS DU TRANSMETTEUR

Les 3 fonctions retournent un booléen indiquant la situation du tampon de transmission.

| Name                               | Description                                                   |
|------------------------------------|---------------------------------------------------------------|
| PLIB_SPI_TransmitBufferIsEmpty     | Returns the current status of the transmit buffer.            |
| PLIB_SPI_TransmitBufferIsFull      | Returns the current transmit buffer status of the SPI module. |
| PLIB_SPI_TransmitUnderRunStatusGet | Returns the current status of the transmit underrun.          |

# 8.3.3. LES FONCTIONS DU RÉCEPTEUR

Les 3 premières fonctions retournent un booléen indiquant la situation du tampon de réception. Il est à remarquer un manque de cohérence dans les noms des fonctions puisque le "Buffer" devient "FIFO" lorsque l'on cherche à savoir s'il est vide.

| Name                           | Description                                              |
|--------------------------------|----------------------------------------------------------|
| PLIB_SPI_ReceiverBufferIsFull  | Returns the current status of the SPI receive buffer.    |
| PLIB_SPI_ReceiverFIFOIsEmpty   | Returns the current status of the SPI receive FIFO.      |
| PLIB_SPI_ReceiverHasOverflowed | Returns the current status of the SPI receiver overflow. |
| PLIB_SPI_ReceiverOverflowClear | Clears the SPI receive overflow flag.                    |



# 8.3.4. LES FONCTIONS DE "DATA TRANSFER"

Le système SPI dispose d'un tampon pour la transmission et la réception, d'où l'existence de fonction pour la gestion de ce tampon (Buffer).

| Name                      | Description                                                                            |
|---------------------------|----------------------------------------------------------------------------------------|
| PLIB_SPI_BufferClear      | Clears the SPI buffer.                                                                 |
| PLIB_SPI_BufferRead       | Returns the SPI buffer value.                                                          |
| PLIB_SPI_BufferAddressGet | Returns the address of the SPIxBUF (Transmit(SPIxTXB) and Receive (SPIxRXB)) register. |
| PLIB_SPI_BufferRead16bit  | Returns 16-bit SPI buffer value.                                                       |
| PLIB_SPI_BufferRead32bit  | Returns 32-bit SPI buffer value.                                                       |
| PLIB_SPI_BufferWrite      | Write the data to the SPI buffer.                                                      |
| PLIB_SPI_BufferWrite16bit | Writes 16-bit data to the SPI buffer.                                                  |
| PLIB_SPI_BufferWrite32bit | Write 32-bit data to the SPI buffer.                                                   |

Nous limitons l'étude aux fonctions de transfert 8 bits ainsi qu'à la fonction d'effacement du buffer.

# 8.3.4.1. LA FONCTION PLIB SPI BUFFERCLEAR

La fonction **PLIB\_SPI\_BufferClear** permet d'effacer le contenu du tampon (émission et réception). Exemple :

PLIB SPI BufferClear(KitSpi1);

# 8.3.4.2. LA FONCTION PLIB\_SPI\_BUFFERREAD

La fonction **PLIB\_SPI\_BufferRead** permet d'obtenir un élément 8 bits du tampon de réception. Son prototype est le suivant :

```
uint8 t PLIB SPI BufferRead (SPI MODULE ID index)
```

Avant de lire il faut s'assurer que le tampon ne soit pas vide!

# 8.3.4.3. LA FONCTION PLIB\_SPI\_BUFFERWRITE

La fonction **PLIB\_SPI\_BufferWrite** permet de déposer un élément 8 bits dans le tampon de transmission. Son prototype est le suivant :

Avant d'écrire, il faut s'assurer que le tampon ne soit pas plein!



# 8.4. ETUDE DU DRIVER SPI FOURNI PAR MHC

L'étude du driver SPI static généré par le MHC) nous fournit un exemple d'utilisation des fonctions de la PLIB SPI.

Cette partie a été réalisée avec les logiciels suivants :

- Harmony v1.06
- MPLABX IDE v3.10
- XC32 v1.40

# **8.4.1.** CONFIGURATION DU DRIVER

Voici la configuration effectuée.



Les modifications apportées sont l'utilisation du CLK\_BUS\_PERIPHERAL\_1 au lieu du 2 et une réduction du SPI Clock Rate à 20 MHz.

Le choix des valeurs pour les sections Clock Mode et Input Phase sont à effectuer en relation avec le composant slave que l'on va utiliser.

# 8.4.2. LA FONCTION DRV\_SPI0\_INITIALIZE

Voici le contenu de la fonction **DRV\_SPI0\_Initialize** 

```
void DRV SPI0 Initialize(void)
{
    PLIB SPI Disable (SPI ID 1);
    PLIB SPI MasterEnable (SPI ID 1);
    PLIB SPI SlaveSelectEnable(SPI ID 1);
    PLIB SPI StopInIdleDisable(SPI ID 1);
    PLIB SPI ClockPolaritySelect (SPI ID 1,
                          SPI CLOCK POLARITY IDLE LOW);
    PLIB SPI OutputDataPhaseSelect (SPI ID 1,
         SPI OUTPUT DATA PHASE ON IDLE TO ACTIVE CLOCK);
    PLIB SPI InputSamplePhaseSelect(SPI ID 1,
                    SPI INPUT SAMPLING PHASE IN MIDDLE);
    PLIB SPI CommunicationWidthSelect(SPI ID 1,
                         SPI COMMUNICATION WIDTH 8BITS);
    PLIB SPI FramedCommunicationDisable (SPI ID 1);
    PLIB SPI FIFOEnable ( SPI ID 1
```



```
PLIB SPI BaudRateSet (SPI ID 1,
                  SYS CLK PeripheralFrequencyGet
                      (CLK BUS PERIPHERAL 1), 20000000);
PLIB SPI Enable (SPI ID 1);
```

Nous reprendrons le principe de la configuration effectuée dans DRV\_SPI0\_Initialize pour l'appliquer à des fonctions d'initialisation spécifiques au DAC et au LM70.

### 8.4.3. LA FONCTION DRV SPIO RECEIVERBUFFERISFULL

La fonction DRV\_SPI0\_ReceiverBufferIsFull nous montre comment tester si le tampon de réception est plein :

```
bool DRV SPIO ReceiverBufferIsFull (void)
    return (PLIB SPI ReceiverBufferIsFull (SPI ID 1));
```

### 8.4.4. LA FONCTION DRV\_SPI0\_TRANSMITTERBUFFERISFULL

La fonction DRV\_SPI0\_TransmitterBufferIsFull nous montre comment tester si le tampon d'émission est plein :

```
bool DRV SPIO TransmitterBufferIsFull (void)
    return (PLIB SPI TransmitBufferIsFull(SPI ID 1));
```

### LA FONCTION DRV\_SPI0\_BUFFERADDWRITEREAD 8.4.5.

La fonction DRV\_SPIO\_BufferAddWriteRead nous montre comment gérer un échange bidirectionnel:

```
int32 t DRV SPIO BufferAddWriteRead (const void * txBuffer,
                            void * rxBuffer, uint32 t size)
{
    bool continueLoop;
    int32 t txcounter = 0;
    int32 t rxcounter = 0;
    uint8 t unitsTxed = 0;
    const uint8 t maxUnits = 16;
    do {
        continueLoop = false;
        unitsTxed = 0;
        if (PLIB SPI TransmitBufferIsEmpty(SPI ID 1))
            while (!PLIB SPI TransmitBufferIsFull(SPI ID 1)
                              && (txcounter < size)
                              && unitsTxed != maxUnits)
            {
                PLIB SPI BufferWrite(SPI ID 1,
                        ((uint8 t*)txBuffer)[txcounter]);
                txcounter++;
                continueLoop = true;
```



```
unitsTxed++;
            }
        }
        while (txcounter != rxcounter)
            while (PLIB SPI ReceiverFIFOIsEmpty(SPI ID 1));
            ((uint8 t*)rxBuffer)[rxcounter] =
                             PLIB SPI BufferRead(SPI ID 1);
            rxcounter++;
            continueLoop = true;
        }
        if (txcounter > rxcounter)
            continueLoop = true;
        }
    }while(continueLoop);
    return txcounter;
}
```

Cette fonction transmet un certain nombre de bytes, avec une limite fixée à 16, puis elle lit ce que le slave a fourni.



# 8.5. RÉALISATION DE L'UTILITAIRE SPI

Les fonctions utilitaires SPI s'inspirent des fonctions SPI du compilateur CCS. L'utilitaire ne s'occupe pas de l'initialisation car elle est spécifique pour chaque composant.

# 8.5.1. RÉALISATION DE LA FONCTION SPI\_WRITE1

Cette fonction effectue l'écriture de 8 bits de données sur le SPI canal 1. L'utilisateur doit gérer le Chip Select avant et après l'utilisation de la fonction.

Cette fonction est fournie dans les fichiers Mc32SpiUtil.c et Mc32SpiUtil.h.

```
void spi_write1( uint8_t Val) {
   bool SpiBusy;

PLIB_SPI_BufferWrite(SPI_ID_1, Val);

// Attente de la fin de la transmission
   do {
        SpiBusy = PLIB_SPI_IsBusy(SPI_ID_1);
   } while (SpiBusy == true);
}
```

La transmission s'effectue en déposant une valeur dans le tampon d'émission (TransmitBuffer) avec la fonction **PLIB\_SPI\_BufferWrite.** 

On aurait pu s'assurer que le tampon d'émission ne soit pas plein avant d'y déposer une nouvelle valeur (à l'aide de la fonction **PLIB\_SPI\_TransmitBufferIsFull**). Ceci n'est toutefois pas nécessaire car on attend la fin de la transmission avant de quitter la fonction. Ainsi, il y aura forcément de la place disponible dans le tampon d'émission au prochain appel de spi\_write1.

Le dépôt d'une valeur dans le tampon d'émission déclenche la transmission. Il est nécessaire d'attendre la fin de cette transmission pour gérer le /CS en synchronisation avec la transmission. Utilisation de la fonction **PLIB\_SPI\_IsBusy.** 

# 8.5.2. RÉALISATION DE LA FONCTION SPI READ1

Cette fonction effectue la lecture de 8 bits de données sur le SPI canal 1. L'utilisateur doit gérer le Chip Select avant et après l'utilisation de la fonction.

Remarque : la fonction a en paramètre la valeur à transmettre et elle retourne la valeur lue.

```
uint8_t spi_read1( uint8_t Val) {
  int SpiBusy;
  uint32_t lu;

PLIB_SPI_BufferWrite(SPI_ID_1, Val);
  // Attends fin transmission
  do {
        SpiBusy = PLIB_SPI_IsBusy(SPI_ID_1);
   } while (SpiBusy == 1);

  // Attend arrivée dans fifo
  while (PLIB_SPI_ReceiverFIFOIsEmpty(SPI_ID_1));
```



```
#ifdef MARKER_READ
    LED3_W = 1;
#endif
    lu = PLIB_SPI_BufferRead(SPI_ID_1);
#ifdef MARKER_READ
    LED3_W = 0;
#endif
    return lu;
}
```

Pour lire, il faut générer des coups d'horloge, ce qui s'effectue avec une fonction d'écriture. Au fur et à mesure que les coups d'horloge sont envoyés, le slave fournit les bits de données.

Il faut attendre la fin de la transmission avec PLIB\_SPI\_IsBusy.

Utilisation de la fonction PLIB\_SPI\_ReceiverFIFOIsEmpty pour attendre tant que le tampon de réception est vide. Ceci est important car si on lit trop vite on manque la valeur.

Si les conditions de départ ne sont pas correctes (tampon vide, pas d'erreur Receive Overflow), il y a risque d'être bloqué ou de ne pas lire la bonne valeur.

L'action PLIB\_SPI\_**BufferClear**(KitSpi1) est nécessaire dans la configuration pour cela.

# 8.5.2.1. SÉQUENCE DES ACTIONS DE LECTURE

Il s'agit d'une séquence de lecture de la température du LM70. La séquence est précédée de la reconfiguration, car l'application gère les 2 composants SPI.

```
SPI_ConfigureLM70();

CS_LM70 = 0;

MSB = spi_read1(0xFF);

LSB = spi_read1(0xFF);

//Fin de transmission
CS LM70 = 1;
```



# 8.5.2.2. OBSERVATION DU MOMENT DE LECTURE DU TAMPON

Pour vérifier à quel moment de la transmission s'effectue la lecture du tampon de réception, nous ajoutons un action sur la LED\_3 pour observer cela.

Canal 1 : CS\_LM70 RD3 broche 78

Canal 2 : SPI-SCK SCK1/RD10 broche 70

Canal 3: LED\_3

Canal 4 : SPI-MISO SDI1/RC4 broche 9



TDS 2024C - 15:07:55 25.05.2016

On peut observer que l'impulsion sur le canal 3 a lieu à la fin de la série de coups d'horloge. Donc les datas sont lues au bon moment, ce qui se confirme par le résultat en température.



# 8.7. PÉRIPHÉRIQUES SPI DU KIT PIC32MX





# 8.8. COMMUNICATION AVEC LE DAC LTC2604

Le dac LTC2604 utilise le module SPI 1.

| Broche du PIC32MX795F512L<br>(100 pin TQFP) | Nom schéma | No broche<br>boitier 100 |
|---------------------------------------------|------------|--------------------------|
| SCK1/IC3/PMCS2/PMA15/RD10                   | SPI-SCK    | 70                       |
| SDO1/OC1/INT0/RD0                           | SPI-MOSI   | 72                       |
| T5CK/ <b>SDI1</b> /RC4                      | SPI-MISO   | 9                        |

# **8.8.1.** Chip Select du LTC2604

Utilisation d'une ligne de port standard (RD4) pour le signal  $\overline{CS}$ 



On dispose aussi de RD9 pour effectuer un reset du composant.

# 8.8.2. CONFIGURATION SPI NÉCESSAIRE AU LTC2604

Au niveau du timing, voici les caractéristiques :

| SYMBOL                 | PARAMETER                                   | CONDITIONS                                                                                     | LTC2604/LTC2614/LTC2624<br>MIN TYP MAX | UNITS    |     |
|------------------------|---------------------------------------------|------------------------------------------------------------------------------------------------|----------------------------------------|----------|-----|
| V <sub>CC</sub> = 2.51 | V to 5.5V                                   |                                                                                                |                                        |          |     |
| t <sub>1</sub>         | SDI Valid to SCK Setup                      |                                                                                                | •                                      | 4        | ns  |
| t <sub>2</sub>         | SDI Valid to SCK Hold                       |                                                                                                | •                                      | 4        | ns  |
| t <sub>3</sub>         | SCK High Time                               |                                                                                                | •                                      | 9        | ns  |
| t <sub>4</sub>         | SCK Low Time                                |                                                                                                | •                                      | 9        | ns  |
| t <sub>5</sub>         | CS/LD Pulse Width                           |                                                                                                | •                                      | 10       | ns  |
| t <sub>6</sub>         | LSB SCK High to CS/LD High                  |                                                                                                | •                                      | 7        | ns  |
| t <sub>7</sub>         | CS/LD Low to SCK High                       |                                                                                                | •                                      | 7        | ns  |
| t <sub>8</sub>         | SDO Propagation Delay from SCK Falling Edge | $C_{LOAD} = 10 pF$<br>$V_{CC} = 4.5 V \text{ to } 5.5 V$<br>$V_{CC} = 2.5 V \text{ to } 5.5 V$ | •                                      | 20<br>45 | ns  |
| t <sub>9</sub>         | CLR Pulse Width                             |                                                                                                |                                        | 20       | ns  |
| t <sub>10</sub>        | CS/LD High to SCK Positive Edge             |                                                                                                | •                                      | 7        | ns  |
|                        | SCK Frequency                               | 50% Duty Cycle                                                                                 | •                                      | 50       | MHz |

Avec la division du PB\_CLOCK par 2, 4, 6, 8, etc., il est possible de travailler à 40 MHz. En pratique, on travaillera à 20 MHz.



Pour le choix du mode d'horloge, le fabricant fournit un diagramme :



On doit donc avoir le flanc montant de l'horloge au milieu des datas.

Ce qui correspond à deux situations : CKP = 1 et CKE = 0 OU CKP = 0 et CKE = 1



# 8.8.2.1. FONCTION DE CONFIGURATION DU SPI POUR LE DAC

Voici le contenu de la fonction SPI\_ConfigureLTC2604 qui est fournie dans le fichier Mc32gestSPiDac.c. Cette fonction reprend les mêmes éléments que la fonction DRV SPI0 Initialize, mais avec l'ajout d'une action PLIB SPI **BufferClear.** 

```
void SPI_ConfigureLTC2604(void)
   PLIB SPI Disable (KitSpi1);
   PLIB SPI BufferClear (KitSpi1);
   PLIB SPI StopInIdleDisable(KitSpi1);
   PLIB SPI PinEnable (KitSpi1, SPI PIN DATA OUT);
   PLIB SPI CommunicationWidthSelect (KitSpi1,
                            SPI COMMUNICATION WIDTH 8BITS);
   // Config SPI clock à 20 MHz
   PLIB SPI BaudRateSet (KitSpi1,
      SYS CLK PeripheralFrequencyGet (CLK BUS PERIPHERAL 1),
                                                  20000000);
   // Config polarité traitement des signaux SPI
   // pour input à confirmer
   // Polarité clock OK
   // Phase output à confirmer
   PLIB SPI InputSamplePhaseSelect(KitSpi1,
                     SPI INPUT SAMPLING PHASE IN MIDDLE );
```



Comme on peut le constater, les fonctions sont assez nombreuses, car elles correspondent aux groupes de bits du registre SPIxCON.

Pour vérifier la configuration on peut lire le registre de configuration.

| [SPI1CON] | 3    | 1    | 30     | )    | 29   |       | 28  |      | 27   |     | 24   |   | 1    | 17 |
|-----------|------|------|--------|------|------|-------|-----|------|------|-----|------|---|------|----|
|           | FRME | N F  | RMSYN  | FRM  | POL  | MSS   | EN  | FRMS | YPW  | FRI | 4CNT | - | SPIE | Œ  |
|           |      | 0    | (      | )    | 0    |       | 0   |      | 0    |     | 000  | - |      | 0  |
|           |      |      |        |      |      |       |     |      |      |     |      |   |      |    |
|           |      | 16 3 | 15     | 13   |      | 12    |     | 11   |      | 10  | 9    |   |      |    |
|           | ENHE | UF ( | ON - 9 | SIDL | DISS | SDO : | MOD | E32  | MODE | 216 | SMP  |   |      |    |
|           |      | 1    | 1 -    | 0    |      | 0     |     | 0    |      | 0   | 0    |   |      |    |
|           |      |      |        |      |      |       |     |      |      |     |      |   |      |    |
|           | 8    |      | 7 6    |      | 5    |       |     | 2    |      | 0   |      |   |      |    |
|           | CKE  | SSE  | N CKP  | MSTE | N -  | STX   | ISE | L SF | XISE | L   |      |   |      |    |
|           | 0    | (    | 0 1    | :    | 1 -  |       | 0   | 0    | (    | 00  |      |   |      |    |

On obtient la valeur 0x00018020, ce qui nous donne CKP = 1 et CKE = 0. Cela correspond au premier choix avec l'horloge qui démarre au niveau haut.

# Si on modifie:



### 8.8.3. DÉTAIL DU SPIXCON

SPIxCON: SPI Control Register (1,2,3)

|   |        |         |        | 0                    |                        |       |                |        |
|---|--------|---------|--------|----------------------|------------------------|-------|----------------|--------|
| Γ | R/W-0  | R/W-0   | R/W-0  | R/W-0                | R/W-0                  | R/W-0 | R/W-0          | R/W-0  |
|   | FRMEN  | FRMSYNC | FRMPOL | MSSEN <sup>(4)</sup> | FRMSYPW <sup>(4)</sup> |       | FRMCNT<2:0>(4) |        |
|   | bit 31 |         |        |                      |                        |       |                | bit 24 |

| r-x    | r-x | r-x | r-x | r-x | r-x | R/W-0 | R/W-0                 |
|--------|-----|-----|-----|-----|-----|-------|-----------------------|
| _      | _   | _   | _   | _   | -   | SPIFE | ENHBUF <sup>(4)</sup> |
| bit 23 |     |     |     |     |     |       | bit 16                |

| R/W-0  | R/W-0 |      |        | R/W-0  | R/W-0  | R/W-0 | R/W-0 |
|--------|-------|------|--------|--------|--------|-------|-------|
| ON     | FRZ   | SIDL | DISSDO | MODE32 | MODE16 | SMP   | CKE   |
| bit 15 |       |      |        |        |        |       | bit 8 |

| R/W-0 | R/W-0 | R/W-0       | r-x | R/W-0 R/W-0 |                       | R/W-0           | R/W-0 |  |
|-------|-------|-------------|-----|-------------|-----------------------|-----------------|-------|--|
| SSEN  | CKP   | CKP MSTEN — |     | STXISE      | L<1:0> <sup>(4)</sup> | SRXISEL<1:0>(4) |       |  |
| bit 7 |       |             |     |             |                       | •               | bit 0 |  |

| Legend:               |                        |                         |                  |  |
|-----------------------|------------------------|-------------------------|------------------|--|
| R = Readable bit      | W = Writable bit       | P = Programmable bit    | r = Reserved bit |  |
| U = Unimplemented bit | -n = Bit Value at POR: | ('0', '1', x = Unknown) |                  |  |

FRMEN: Framed SPI Support bit bit 31

1 = Framed SPI support is enabled (SSx pin used as FSYNC input/output)

0 = Framed SPI support is disabled

bit 30 FRMSYNC: Frame Sync Pulse Direction Control on SSx pin bit (Framed SPI mode only)

> 1 = Frame sync pulse input (Slave mode) 0 = Frame sync pulse output (Master mode)

bit 29 FRMPOL: Frame Sync Polarity bit (Framed SPI mode only)

1 = Frame pulse is active-high 0 = Frame pulse is active-low

MSSEN: Master Mode Slave Select Enable bit(4) bit 28

> 1 = Slave select SPI support enabled. The SS pin is automatically driven during transmission in Master mode. Polarity is determined by the FRMPOL bit.

0 = Slave select SPI support is disabled.

bit 27 FRMSYPW: Frame Sync Pulse Width bit (4)

1 = Frame sync pulse is one character wide

0 = Frame sync pulse is one clock wide

# Register 23-1: SPIxCON: SPI Control Register (1,2,3) (Continued)

FRMCNT<2:0>: Frame Sync Pulse Counter bits. Controls the number of data characters transmitted per pulse.<sup>(4)</sup> bit 26-24

111 = Reserved; do not use

110 = Reserved; do not use

101 = Generate a frame sync pulse on every 32 data characters

100 = Generate a frame sync pulse on every 16 data characters

011 = Generate a frame sync pulse on every 8 data characters

010 = Generate a frame sync pulse on every 4 data characters

001 = Generate a frame sync pulse on every 2 data characters

000 = Generate a frame sync pulse on every data character This bit is only valid in FRAMED\_SYNC mode.

bit 23-18 Reserved: Write '0'; ignore read

bit 17 SPIFE: Frame Sync Pulse Edge Select bit (Framed SPI mode only)

1 = Frame synchronization pulse coincides with the first bit clock

0 = Frame synchronization pulse precedes the first bit clock

ENHBUF: Enhanced Buffer Enable bit(4) bit 16

1 = Enhanced Buffer mode is enabled

0 = Enhanced Buffer mode is disabled



bit 15 ON: SPI Peripheral On bit

1 = SPI Peripheral is enabled

0 = SPI Peripheral is disabled

Note: When using the 1:1 PBCLK divisor, the user's software should not read or write the peripheral's SFRs in the SYSCLK cycle immediately following the instruction that clears the module's ON bit.

bit 14 FRZ: Freeze in Debug Exception Mode bit

1 = Freeze operation when CPU enters Debug Exception mode

0 = Continue operation when CPU enters Debug Exception mode

Note: FRZ is writable in Debug Exception mode only, it is forced to '0' in Normal mode.

bit 13 SIDL: Stop in Idle Mode bit

1 = Discontinue operation when CPU enters in Idle mode

0 = Continue operation in Idle mode

bit 12 DISSDO: Disable SDOx pin bit

1 = SDOx pin is not used by the module. Pin is controlled by associated PORT register

0 = SDOx pin is controlled by the module

bit 11-10 MODE<32,16>: 32/16-Bit Communication Select bits

1x = 32-bit data width 01 = 16-bit data width

00 = 8-bit data width

Note 1: This register has an associated Clear register (SPIxCONCLR) at an offset of 0x4 bytes. Writing a '1' to any bit position in the Clear register will clear valid bits in the associated register. Reads from the Clear register should be ignored.

- This register has an associated Set register (SPIxCONSET) at an offset of 0x8 bytes. Writing a '1' to any bit position in the Set register will set valid bits in the associated register. Reads from the Set register should be ignored.
- 3: This register has an associated Invert register (SPIxCONINV) at an offset of 0xC bytes. Writing a '1' to any bit position in the Invert register will invert valid bits in the associated register. Reads from the Invert register should be ignored.
- 4: These bits are not available on all devices. Refer to the specific device data sheet for availability.

# Register 23-1: SPIxCON: SPI Control Register (1,2,3) (Continued)

bit 9 SMP: SPI Data Input Sample Phase bit

Master mode (MSTEN = 1):

1 = Input data sampled at end of data output time

0 = Input data sampled at middle of data output time

Slave mode (MSTEN = 0):

SMP value is ignored when SPI is used in Slave mode. The module always uses SMP = 0.

bit 8 CKE: SPI Clock Edge Select bit

1 = Serial output data changes on transition from active clock state to Idle clock state (see CKP bit)

0 = Serial output data changes on transition from Idle clock state to active clock state (see CKP bit)

Note: The CKE bit is not used in the Framed SPI mode. The user should program this bit to '0' for the Framed SPI mode (FRMEN = 1).

bit 7 SSEN: Slave Select Enable (Slave mode) bit

1 = SSx pin used for Slave mode

0 = SSx pin not used for Slave mode, pin controlled by port function.

bit 6 CKP: Clock Polarity Select bit

1 = Idle state for clock is a high level; active state is a low level

0 = Idle state for clock is a low level; active state is a high level

bit 5 MSTEN: Master Mode Enable bit

1 = Master mode

0 = Slave mode

bit 4 Reserved: Write '0'; ignore read

bit 3-2 STXISEL<1:0>: SPI Transmit Buffer Empty Interrupt Mode bits<sup>(4)</sup>

11 = SPI\_TBE\_EVENT is set when the buffer is not full (has one or more empty elements)

10 = SPI\_TBE\_EVENT is set when the buffer is empty by one-half or more

01 = SPI\_TBE\_EVENT is set when the buffer is completely empty

00 = SPI\_TBE\_EVENT is set when the last transfer is shifted out of SPISR and transmit operations are complete

bit 1-0 RTXISEL<1:0>: SPI Receive Buffer Full Interrupt Mode bits<sup>(4)</sup>

11 = SPI\_RBF\_EVENT is set when the buffer is full

10 = SPI\_RBF\_EVENT is set when the buffer is full by one-half or more

01 = SPI\_RBF\_EVENT is set when the buffer is not empty

00 = SPI\_RBF\_EVENT is set when the last word in the receive buffer is read (i.e., buffer is empty)



# 8.8.4. SÉLECTION DE LA FRÉQUENCE DE SCK

Voici la formule qui détermine la fréquence de SCK en fonction de la fréquence de PB\_CLOCK et du registre SPIxBRG.

$$F_{SCK} = \frac{F_{PB}}{2 \cdot (SPIxBRG + 1)}$$

Ce qui donne des divisions possibles par 2, 4, 6, 8, etc.

Avec PB\_CLOCK = 80 MHz et SPIxBRG = 0, la fréquence maximum de SCK sera de 40 MHZ.

Lors de la configuration, nous avons établi :

```
PLIB_SPI_BaudRateSet(KitSpi1,

SYS_CLK_PeripheralFrequencyGet(CLK_BUS_PERIPHERAL_1),

20000000:
```

Pour obtenir 20 MHz, il faut diviser par 4 donc SPIxBRG doit valoir 1. Ce que nous vérifions en lisant la valeur de SPI1BRG.

```
PLIB_SPI_Enable(KitSpi1);

[SPI1BRG] 0

// Contrôle 1
ConfigReg = S

BaudReg = SPI1BRG;
```

# 8.8.5. Initialisation du LTC2604

Voici la fonction d'initialisation complète avec le reset du LTC2604.

```
void SPI_InitLTC2604(void)
{
    //Initialisation SPI DAC
    CS_DAC = 1;
    // Impulsion reset du DAC
    DAC_CLEAR = 0;
    delay_us(500);
    DAC_CLEAR = 1;
    // LTC2604 MAX 50 MHz choix 20 MHz
    SPI_ConfigureLTC2604();
}
```



# 8.8.6. ECRITURE D'UNE VALEUR SUR LE LTC2604

Voici la fonction d'écriture sur le LTC2604. Dans le but de partager le bus SPI avec un autre composant, la fonction d'écriture reconfigure le SPI avant l'action.

```
// Envoi d'une valeur sur le DAC LTC2604
// Avec reconfiguration du SPI
// Indication du canal 0 à 3
void SPI_CfgWriteToDac(uint8_t NoCh, uint16 t DacVal)
   uint8 t MSB;
   uint8 t LSB;
   // Reconfiguration du SPI
   SPI ConfigureLTC2604();
   //Sélection du canal
   //3 -> Set and Update, 0/1/2/3 Sélection canal A/B/C/D,
                                  F tous canaux
  NoCh = NoCh + 0x30;
  MSB = DacVal >> 8;
   LSB = DacVal;
   CS DAC = 0;
   spi write1(NoCh);
   spi write1(MSB);
   spi write1(LSB);
   // Fin de transmission
   CS DAC = 1;
} // SPI CfgWriteToDac
```

Ce qui correspond à l'indication du fabricant :

COMMAND ADDRESS DATA (16 BITS)

C3 C2 C1 C0 XA3 XA2 XA1 XA0 XD15 XD14 XD13 XD12 XD11 XD10 X D9 X D8 X D7 X D6 X D5 X D4 X D3 X D2 X D1 X D0

MSB

LSB

2004 TELSH
200



# 8.8.7. OBSERVATION DES SIGNAUX DU DAC

Canal 1 : CS\_DAC RD4 broche 81

Canal 2 : SPI-SCK SCK1/RD10 broche 70

Canal 3 : SPI-MOSI SDO1/RD0 broche 72

Canal 4 : sortie du DAC 0



TDS 2024C - 17:53:14 25:05:2016

On observe les 3 salves de 8 coups d'horloge. La valeur envoyée est 0x81 pour le MSB et 0x5A sur le LSB (3ème data).

# 8.8.7.1. DÉTAIL DU 3ÈME OCTET

La valeur du 3<sup>ème</sup> octet (LSB) est imposée à 0x5A. On peut donc vérifier que le flanc montant du clock est au milieu du data. Au 1<sup>er</sup> flanc montant on a bien un 0 et au 5<sup>ème</sup> flanc un 1.



TDS 2024C - 17:58:31 25.05.2016

Les 4 périodes d'horloge valent 200 ns, ce qui nous donne 50 ns pour une période donc une fréquence de 20 MHz comme prévu.

Le fait que l'horloge démarre sur un état haut ne pose pas de problème.



# 8.8.7.2. SIGNAL SUR LE DAC

Si on transmet des valeurs qui se suivent, comme par exemple :

```
SPI_CfgWriteToDac(0, DacVal);
DacVal += 2048;
```

On peut observer les changements de valeur du signal, ce qui prouve la bonne interprétation par le DAC.



TDS 2024C - 17:35:08 26.05.2016



# 8.10. COMMUNICATION AVEC LE LM70

Nous allons illustrer principalement l'utilisation des fonctions de lecture du SPI par la communication avec le composant LM70 qui est un capteur de température.

# 8.10.1. CONNEXIONS ENTRE LE PIC32 ET LE LM70

Utilisation du SPI1 et d'une ligne de port standard (RD3) pour le signal CS

| Broche du PIC32MX795F512L<br>(100 pin TQFP)    | Nom schéma  | No broche<br>boitier 100 |
|------------------------------------------------|-------------|--------------------------|
| SCK1/IC3/PMCS2/PMA15/RD10 70 >>SPI-SCK         | SPI-SCK     | 70                       |
| SDO1/OC1/INT0/RD0 72 >>SPI-MOSI                | SPI-MOSI    | 72                       |
| SPI-MISO )> 9 T4CK/AC2RO/RC3-<br>T5CK/SDI1/RC4 | SPI-MISO    | 9                        |
| OC4/RD3 78 SPI-CS_Temp                         | SPI-CS_Temp | 78                       |

Comme le LM70 ne possède qu'une ligne bidirectionnelle, il est nécessaire d'utiliser une résistance pour combiner les lignes SPI-MOSI et SPI-MISO SDO du PIC.



8.10.2. CONFIGURATION SPI NÉCESSAIRE AU LM70

# Le fabricant du LM70 indique :

The LM70 operates as a slave and is compatible with SPI or MICROWIRE bus specifications. Data is clocked out on the falling edge of the serial clock (SC), while data is clocked in on the rising edge of SC.

Pour la lecture du LM70 par le maître, les données sont fournies par l'esclave au flanc descendant de SCK, tandis que pour l'écriture par le maître, les datas sont lues au flanc montant de l'horloge. Ce qui correspond à CKP = 0 et CKE = 0.





# 8.10.2.1. FONCTION DE CONFIGURATION DU SPI POUR LE LM70

Voici le contenu de la fonction SPI\_ConfigureLM70 qui est fournie dans les fichiers Mc32GestSpiLM70.h et Mc32GestSpiLM70.c.

```
void SPI ConfigureLM70 (void)
   PLIB SPI Disable (KitSpi1);
   PLIB SPI BufferClear (KitSpi1);
   PLIB SPI StopInIdleDisable(KitSpi1);
   PLIB SPI PinEnable (KitSpi1, SPI PIN DATA OUT);
   PLIB SPI CommunicationWidthSelect (KitSpi1,
                         SPI COMMUNICATION WIDTH 8BITS);
   // LM70 MAX 6.25 MHz choix 5 MHz
   PLIB SPI BaudRateSet (KitSpil,
      SYS CLK PeripheralFrequencyGet (CLK BUS PERIPHERAL 1),
   // Config polarité traitement des signaux SPI
   // pour input à confirmer
   // Polarité clock OK
   // Phase output à confirmer
   PLIB SPI InputSamplePhaseSelect(KitSpi1,
                      SPI INPUT SAMPLING PHASE IN MIDDLE );
   PLIB SPI ClockPolaritySelect (KitSpi1,
                               SPI CLOCK POLARITY IDLE LOW);
   PLIB SPI OutputDataPhaseSelect (KitSpi1,
            SPI OUTPUT DATA PHASE ON IDLE TO ACTIVE CLOCK);
   PLIB SPI MasterEnable (KitSpi1);
   PLIB SPI FramedCommunicationDisable(KitSpi1);
   PLIB SPI FIFOEnable (KitSpi1); // Enhenced buffer mode
   PLIB SPI Enable (KitSpi1);
   // Contrôle de la configuration
   ConfigReg = SPI1CON;
   BaudReg = SPI1BRG;
}
```

# 8.10.2.2. VÉRIFICATION DE LA CONFIGURATION AVEC SPI1CON

| [SPI1CON] | 3:    | 1     | 30    | )      | 29  |     | 28  |      | 27    |     | 24   |   | 1     | 7 |
|-----------|-------|-------|-------|--------|-----|-----|-----|------|-------|-----|------|---|-------|---|
|           | FRME  | N FRI | 4SYNC | FRME   | OL  | MSS | EN  | FRMS | SYPW  | FRI | MCNT | - | SPIFE | E |
|           | (     | 0     | 0     | )      | 0   |     | 0   |      | 0     |     | 000  | - | (     | 0 |
|           |       |       |       |        |     |     |     |      |       |     |      |   |       |   |
|           | :     | 16 15 | 5     | 13     |     | 12  |     | 11   |       | 10  | 9    |   |       |   |
|           | ENHBU | UF ON | 1 - 5 | SIDL I | ISS | SDO | MOD | E32  | MODE  | 216 | SMP  |   |       |   |
|           |       | 1 1   | L -   | 0      |     | 0   |     | 0    |       | 0   | 0    |   |       |   |
|           |       |       |       |        |     |     |     |      |       |     |      |   |       |   |
|           | 8     | 7     | 6     | 5      | 5   |     |     | 2    |       | 0   |      |   |       |   |
|           | CKE S | SSEN  | CKP   | MSTEN  | 1 - | STX | ISE | L SE | XXISE | EL  |      |   |       |   |
|           | 0     | 0     | 0     | 1      | -   |     | 0   | 0    | (     | 00  |      |   |       |   |

On a bien confirmation de CKE = 0 et CKP = 0.



# Serial Bus Digital Switching Characteristics

Unless otherwise noted, these specifications apply for V<sup>+</sup> = 2.65V to 3.6V for the LM70-3 and V<sup>+</sup> = 4.5V to 5.5V for the LM70-5,  $C_L$  (load capacitance) on output lines = 100 pF unless otherwise specified. **Boldface limits apply for T<sub>A</sub> = T<sub>J</sub> = T<sub>MIN</sub>** to  $T_{MAX}$ ; all other limits  $T_A = T_J = +25^{\circ}C$ , unless otherwise noted.

| Parameter                                   | Conditions                                                                                                                                                                                                | Typical<br>(Note 7)                                                                                                                                                                                       | Limits<br>(Note 8)                                                                                                                                                                                                                                 | Units<br>(Limit)  |
|---------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------|
| SC (Clock) Period                           |                                                                                                                                                                                                           |                                                                                                                                                                                                           | 0.16<br>DC                                                                                                                                                                                                                                         | μs (min)<br>(max) |
| CS Low to SC (Clock) High Set-Up Time       |                                                                                                                                                                                                           |                                                                                                                                                                                                           | 100                                                                                                                                                                                                                                                | ns (min)          |
| CS Low to Data Out (SO) Delay               |                                                                                                                                                                                                           |                                                                                                                                                                                                           | 70                                                                                                                                                                                                                                                 | ns (max)          |
| SC (Clock) Low to Data Out (SO) Delay       |                                                                                                                                                                                                           |                                                                                                                                                                                                           | 70                                                                                                                                                                                                                                                 | ns (max)          |
| CS High to Data Out (SO) TRI-STATE          |                                                                                                                                                                                                           |                                                                                                                                                                                                           | 200                                                                                                                                                                                                                                                | ns (min)          |
| SC (Clock) High to Data In (SI) Hold Time   |                                                                                                                                                                                                           |                                                                                                                                                                                                           | 60                                                                                                                                                                                                                                                 | ns (min)          |
| Data In (SI) Set-Up Time to SC (Clock) High |                                                                                                                                                                                                           |                                                                                                                                                                                                           | 30                                                                                                                                                                                                                                                 | ns (min)          |
|                                             | SC (Clock) Period  CS Low to SC (Clock) High Set-Up Time CS Low to Data Out (SO) Delay SC (Clock) Low to Data Out (SO) Delay CS High to Data Out (SO) TRI-STATE SC (Clock) High to Data In (SI) Hold Time | SC (Clock) Period  CS Low to SC (Clock) High Set-Up Time CS Low to Data Out (SO) Delay SC (Clock) Low to Data Out (SO) Delay CS High to Data Out (SO) TRI-STATE SC (Clock) High to Data In (SI) Hold Time | SC (Clock) Period (Note 7)  SC (Clock) Period (Note 7)  CS Low to SC (Clock) High Set-Up Time  CS Low to Data Out (SO) Delay  SC (Clock) Low to Data Out (SO) Delay  CS High to Data Out (SO) TRI-STATE  SC (Clock) High to Data In (SI) Hold Time | SC (Clock) Period |

# **Timing Diagrams**



Le minimum de 0,16 us pour le serial clock correspond à un max de 6.25 MHz. Avec la division par 16 on obtient si la fréquence d'oscillateur est de 80 MHz, 80/16 = 5 MHz.

Lors de la configuration nous avons établi :

Pour obtenir 5 MHz, il faut diviser par 16 donc SPIxBRG doit valoir 7, car 2(7+1) = 16. Ce que nous vérifions en lisant la valeur de SPI1BRG.

```
61
62 // Contrôle le SPI1BRG n
63 ConfigReg = SP 000000111

BaudReg = SPI1BRG; ....ce qui correspond bien à 7.
```

# 8.10.3. Initialisation du LM70

La fonction **SPI\_InitLM70** effectue la configuration du SPI pour le LM70 et effectue une séquence permettant de configurer les registres du LM70.

```
void SPI_InitLM70 (void) {
   SPI_ConfigureLM70();
   // action de configuration
   CS_LM70 = 0;
   spi_read1(0xFF);
   spi_read1(0xFF);
   spi_read1(0); // pour écrire 0
   spi_read1(0); // pour écrire 0
   //Fin de transmission
   CS_LM70 = 1;
} // SPI InitLM70
```



# **8.10.4. LECTURE DU LM70**

Pour réaliser une lecture, il faut respecter la séquence ci-dessous :



b) Reading Continuous Conversion - Two Eight-Bit Frames

On remarque que le signal *CS* doit être activé au début de l'action et désactivé à la fin de l'action. La lecture s'effectue en deux trains de 8 bits, ce qui correspond bien à l'usage du MSSP.

# 8.10.5. LA FONCTION SPI\_READRAWTEMPLM70

La fonction SPI\_ReadRawTempLM70 effectue la lecture en appelant 2 fois la fonction **spi\_read1**. On obtient d'abord le poids fort et ensuite le poids faible. La valeurs des 2 octets envoyés n'a pas d'importance.

```
// Lecture du registre de température du LM70
// Version sans reconfiguration
int16 t SPI ReadRawTempLM70(void)
   //Déclaration des variables
   uint8 t MSB;
   uint8 t LSB;
   int16 t RawTemp;
   CS LM70 = 0;
  MSB = spi read1(0xFF);
   LSB = spi read1(0xFF);
   //Fin de transmission
   CS LM70 = 1;
   RawTemp = MSB;
  RawTemp = RawTemp << 8;
  RawTemp = RawTemp | LSB;
   return RawTemp;
} // SPI ReadRawTempLM70
```

Pour obtenir la température il faut combiner le msb et le lsb et manipuler la valeur en tenant compte des détails du registre de température :

# 1.5.2 TEMPERATURE REGISTER

(Read Only):

| III WAS A STATE OF THE PARTY OF | The second second second |       |       |       |       |       |       |       |       |     |    |    |    |    |    |
|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------|-------|-------|-------|-------|-------|-------|-------|-------|-----|----|----|----|----|----|
| D15                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            | D14                      | D13   | D12   | D11   | D10   | D9    | D8    | D7    | D6    | D5  | D4 | D3 | D2 | D1 | D0 |
| MSB                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            | Bit 9                    | Bit 8 | Bit 7 | Bit 6 | Bit 5 | Bit 4 | Bit 3 | Bit 2 | Bit 1 | LSB | 1  | 1  | 1  | Х  | X  |

D0-D1: Undefined. TRI-STATE will be output on SI/0.

D2-D4: Always set high.

D5-D15: Temperature Data. One LSB =  $0.25^{\circ}$ C. Two's complement format.



# 8.10.6. ECRITURE VERS LE LM70

Il y a écriture uniquement pour la configuration du mode de Shutdown. Il y a 2 cycles de lecture avant les 2 cycles d'écriture. C'est ce que nous déduisons du diagramme cidessous :



On remarque que le signal  $\overline{CS}$  doit être activé au début de l'action et désactivé à la fin.

# 1.5.1 CONFIGURATION REGISTER

(Selects shutdown or continuous conversion modes):

(Write Only):

| D15 | D14 | D13 | D12 | D11 | D10 | D9 | D8 | D7 | D6 | D5 | D4    | D3   | D2 | D1 | D0 |
|-----|-----|-----|-----|-----|-----|----|----|----|----|----|-------|------|----|----|----|
| X   | X   | Х   | X   | X   | Х   | X  | X  |    | 2  |    | Shute | down | 3  | 3  |    |

D0-D15 set to XX FF hex enables shutdown mode.

D0-D15 set to XX 00 hex enables continuous conversion mode.

Note: setting D0-D15 to any other values may place the LM70 into a manufacturer's test mode, upon which the LM70

will stop responding as described. These test modes are to be used for National Semiconductor production testing only. See Section 1.2 Serial Bus Interface for a complete discussion.

L'écriture s'adresse uniquement au registre de configuration. Il faut écrire XX 00 pour obtenir le mode continu. Dans l'exemple d'écriture on utilise 00 00.

# 8.10.6.1. EXEMPLE D'ÉCRITURE

Dans l'exemple ci-dessous, il y a écriture d'une valeur 16 bits valant 0. Pour éviter des problèmes avec le buffer de réception, il faut utiliser la fonction de lecture aussi pour écrire les 0. D'où :

```
// action de configuration
CS_LM70 = 0;
spi_read1(0xFF);
spi_read1(0xFF);
spi_read1(0); // pour écrire 0
spi_read1(0); // pour écrire 0
//Fin de transmission
CS_LM70 = 1;
```



# 8.10.7. OBSERVATION DES SIGNAUX DU LM70

Voici la vue d'ensemble lors de la lecture du registre de température :

Canal 1 : CS\_LM70 RD3 broche 78

Canal 2 : SPI-SCK SCK1/RD10 broche 70

Canal 3 : SPI-MOSI SDO1/RD0 broche 72

Canal 4 : SPI-MISO SDI1/RC4 broche 9



On peut observer que le MOSI (canal 3) est à l'état haut durant l'action. Le LM70 fournit une valeur dès réception des coups d'horloge.



# **8.10.7.1. V**UE DE DÉTAIL

Voici la vue de détail, observation du traitement du 1<sup>er</sup> octet (MSB, le plus stable).



La fréquence d'horloge est bien de 5 MHz car 4 périodes donnent 800 ns et pour 5 MHz la période est de 200 ns.

La valeur de l'octet de poids fort transmis est 0x0D, soit 0000'1101, ce que le master peut lire au flancs montants de l'horloge. Le LM70 fournit un nouveau bit lors de chaque flanc descendant de l'horloge (sauf le tout premier bit, qui est fourni dès le /CS bas). Au 1<sup>er</sup> flanc descendant, on a donc le 2<sup>ème</sup> bit qui est mis en sortie du LM70. Lors du dernier flanc descendant de l'horloge, le data passe déjà à l'octet suivant.

La figure datasheet nous confirme tout cela:





# 8.11. UTILISATION COMBINÉE DES 2 SLAVES SPI

Pour permettre d'utiliser sur le même bus SPI du LM70 et du LTC2604, il est nécessaire de reconfigurer le SPI avant chaque transaction. Cela permet donc d'utiliser des fréquences d'horloge différentes et des modes d'horloge différents.

Au niveau de l'initialisation on exécutera une fois la fonction d'initialisation du LTC2604 pour s'assurer de son Reset.

# 8.11.1. FONCTION LECTURE LM70 AVEC RECONFIGURATION

Lecture du registre de température du LM70, version avec reconfiguration pour partage du bus SPI avec un autre composant.

```
// Lecture du registre de température du LM70
// Version avec reconfiguration
int16 t SPI CfgReadRawTempLM70(void)
   uint8 t MSB;
   uint8 t LSB;
   int16 t RawTemp;
   SPI ConfigureLM70();
   CS LM70 = 0;
  MSB = spi read1(0xFF);
   LSB = spi read1(0xFF);
   //Fin de transmission
   CS LM70 = 1;
  RawTemp = MSB;
  RawTemp = RawTemp << 8;</pre>
  RawTemp = RawTemp | LSB;
   return RawTemp;
} // SPI CfgReadRawTempLM70
```



# 8.11.2. FONCTION ÉCRITURE LTC2604 AVEC RECONFIGURATION

Cette fonction appelle la fonction de reconfiguration avant d'effectuer la transaction.

```
void SPI CfgWriteToDac(uint8 t NoCh, uint16 t DacVal)
   //Déclaration des variables
   uint8 t MSB;
   uint8 t LSB;
   // Reconfiguration du SPI
   SPI ConfigureLTC2604();
   //Sélection du canal
   //3 -> Set and Update, 0/1/2/3 Sélection canal A/B/C/D,
   //
                                             F tous canaux
  NoCh = NoCh + 0x30;
   // Selon canal
  MSB = DacVal >> 8;
   LSB = DacVal;
  CS DAC = 0;
   spi write1(NoCh);
   spi write1(MSB);
   spi write1(LSB);
   // Fin de transmission
   CS DAC = 1;
} // SPI CfgWriteToDac
```

# 8.11.3. UTILISATION ET OBTENTION DES RÉSULTATS

Dans cet exemple, on transmet les valeurs au DAC dans une routine d'interruption. Pour garantir que la séquence de lecture du LM70 ne soit pas interrompue par le dialogue du DAC, on la place également dans l'interruption du DAC en utilisant un compteur.

# 8.11.3.1. CONTENU DE LA RÉPONSE À L'INTERRUPTION DU TIMER1

Voici le contenu de la réponse à l'interruption du Timer1.



```
PLIB INT SourceFlagClear(INT ID 0, INT SOURCE TIMER 1);
    count++;
    // DacVal = 0x815A; // pour observation signaux
    SPI CfgWriteToDac(0, DacVal);
    DacVal += 2048;
    if (count >= 5000) {
        count = 0;
        delay us(5); // pour séparation des signaux
        RawTemp = SPI CfgReadRawTempLM70();
        LM70 ConvRawToDeg( RawTemp, &TempLm70);
        APP UpdateTemp (RawTemp, TempLm70);
        APP UpdateState (APP STATE SERVICE TASKS);
} // end ISR
```

### 8.11.3.2. AFFICHAGE TEMPÉRATURE DANS L'APPLICATION

L'affichage est réalisé dans l'application. Les fonctions de mise à jour appelées depuis l'interruption permettent un accès propre aux variables de l'application.

```
APP DATA appData;
int16 t APP RawTemp = 225;
float APP TempLm70 = 21.7;
case APP STATE SERVICE TASKS:
     BSP LEDToggle (BSP LED 2);
     // Affichage temperature du LM70
     lcd gotoxy(1,3);
     printf lcd("RawTemp = %08X", APP RawTemp);
     lcd gotoxy(1,4);
     printf lcd("Temp = %6.1f", APP TempLm70);
     appData.state = APP STATE WAIT;
break;
```

### 8.11.3.3. PROBLÈME AVEC LES VARIABLES FLOAT DANS L'INTERRUPTION

Dans l'interruption, nous avons déclaré les 2 variables pour le LM70 de la manière suivante:

```
static int16 t RawTemp;
static float TempLm70;
```

En écrivant:

```
TempLm70 = LM70 ConvRawToDeg( RawTemp);
```

Le résultat dans TempLm70 était totalement incohérent.

En modifiant en:

```
LM70 ConvRawToDeg( RawTemp, &TempLm70);
Le résultat est correct.
```

😊 La manipulation d'une variable float implique un appel à une fonction système, il semble que cela soit la cause du problème, sans pour autant l'expliquer.



En utilisant le passage par référence, ce problème disparait.

# 8.11.4. REMARQUE SUR LES RECONFIGURATIONS

La reconfiguration avant chaque communication avec un des périphériques SPI permet de passer d'une horloge à 20 MHz pour le LTC2604 à une horloge à 5 MHz pour le LM70. Les modes d'horloge sont aussi adaptés.

La reconfiguration présente l'avantage d'effacer les erreurs, ce qui permet d'assurer une lecture propre du LM70, alors que l'on a utilisé la fonction d'écriture pour le LTC2604.

# 8.12. FICHIERS À DISPOSITION

Les librairies permettant l'utilisation des composants SPI sont fournies dans le répertoire suivant :

...\Maitres-Eleves\SLO\Modules\SL229\_MINF\PIC32MX\_Utilitaires(PlibHarmony)\SPI



# 8.13. CONCLUSION

Ce document devrait permettre, en s'inspirant des principes utilisés, de s'adapter à la gestion par le bus SPI d'autres composants que ceux présentés.

Il s'agira à chaque fois d'adapter la configuration SPI (notamment la fréquence du clock, sa polarité et le nombre de tranches de 8 bits à lire ou à écrire). Une étude détaillée des datasheets est indispensable.



# 8.14. HISTORIQUE DES VERSIONS

# 8.14.1. **VERSION 1.0 MAI 2014**

Transformation du chapitre 13 (théorie PIC18) pour obtenir la 1<sup>ère</sup> version de ce chapitre.

# 8.14.2. **VERSION 1.1 MAI 2014**

Remplacement des extraits de schéma PIC18 pour le LM70. Complément info sur le SPI1CON. Ajout observation des signaux du LM70.

# 8.14.3. Version 1.5 mars 2015

Redevient un chapitre de théorie. Adaptation à la PLIB\_SPI de Harmony V 1.00 et description d'une partie des fonctions.

# 8.14.4. VERSION 1.7 MAI 2016

Version 1.7 pour compatibilité avec l'ensemble des modules. Adaptation à la PLIB\_SPI de Harmony V 1.06 et ajout de l'étude du driver SPI fourni par le MHC. Correction de la fonction spi\_read et contrôle du bon fonctionnement.

# 8.14.5. VERSION 1.8 MARS 2017

Relecture générale par SCA.

# 8.14.6. Version 1.9 Février 2018

Ajouts documents de référence. Corrections mineures.

# 8.14.7. Version 1.91 mars 2019

Correction mineure fonction spi\_write1.