## **Serial Interfaces**

Lecture 11

Josh Brake
Harvey Mudd College

### **Outline**

- Serial Interfaces Overview
  - Advantages over parallel
  - Major considerations
  - Overview of protocols
- Serial Peripheral Interface
  - Description
  - MCU configuration
- DS1722 SPI temperature sensor
  - Datasheet overview
- CMSIS

### **Learning Objectives**

By the end of this lecture you should be able to...

- See how the SPI peripheral works on the STM32L432KC
- See how to verify the output using a logic analyzer

## **Serial Interfaces Overview**

### **Motivation**

How can we interface a peripheral?

Imagine transmitting a character on a keyboard.

Capital J in ASCII is  $74_{10} = 01001010_2$ 



## What if we repackage data in a stream?

- Multiplexing in time
- To send N bits, we only need 2 lines (CLK + Data) instead of 9
- Price we pay is time but often worth it.



## Serial Peripheral Interface (SPI)

- Developed in the mid-1980s by Motorola
- Used to interface with many peripherals like memory (SD cards, flash), displays, sensors (accelerometers, gyroscopes, temperature sensors, ADCs and DACs).
- Four-wire, synchronous serial bus



SCLK: Serial clock

MOSI: Master Out Slave In

MISO: Master In Slave Out

CE/CS/nCE/nCS: Chip select/enable

## SPI Block Diagram on STM32L432KC



Figure 419. SPI block diagram

## **SPI Block Diagram**



Figure 420. Full-duplex single master/ single slave application

### **SPI Hardware NSS Management**



## **Example SPI Traces**





### **SPI Data Frame Sizes**

Figure 427. Data alignment when data length is not equal to 8-bit or 16-bit



The minimum data length is 4 bits. If a data length of less than 4 bits is selected, it is forced to an 8-bit data frame size.

### **SPI Configuration**

#### Configuration of SPI

The configuration procedure is almost the same for master and slave. For specific mode setups, follow the dedicated sections. When a standard communication is to be initialized, perform these steps:

- 1. Write proper GPIO registers: Configure GPIO for MOSI, MISO and SCK pins.
- 2. Write to the SPI\_CR1 register:
  - a) Configure the serial clock baud rate using the BR[2:0] bits (Note: 4).
  - b) Configure the CPOL and CPHA bits combination to define one of the four relationships between the data transfer and the serial clock (CPHA must be cleared in NSSP mode). (Note: 2 - except the case when CRC is enabled at TI mode).
  - c) Select simplex or half-duplex mode by configuring RXONLY or BIDIMODE and BIDIOE (RXONLY and BIDIMODE can't be set at the same time).
  - d) Configure the LSBFIRST bit to define the frame format (Note: 2).
  - e) Configure the CRCL and CRCEN bits if CRC is needed (while SCK clock signal is at idle state).
  - f) Configure SSM and SSI (Notes: 2 & 3).
  - g) Configure the MSTR bit (in multimaster NSS configuration, avoid conflict state on NSS if master is configured to prevent MODF error).
- 3. Write to SPI\_CR2 register:
  - a) Configure the DS[3:0] bits to select the data length for the transfer.
  - b) Configure SSOE (Notes: 1 & 2 & 3).
  - c) Set the FRF bit if the TI protocol is required (keep NSSP bit cleared in TI mode).
  - Set the NSSP bit if the NSS pulse mode between two data units is required (keep CHPA and TI bits cleared in NSSP mode).
  - e) Configure the FRXTH bit. The RXFIFO threshold must be aligned to the read access size for the SPIx\_DR register.
  - f) Initialize LDMA\_TX and LDMA\_RX bits if DMA is used in packed mode.
- 4. Write to SPI\_CRCPR register: Configure the CRC polynomial if needed.
- Write proper DMA registers: Configure DMA streams dedicated for SPI Tx and Rx in DMA registers if the DMA streams are used.

### **SPI Clock Polarity and Phase**

- Clock polarity (CPOL) refers to the state of the clock line at idle
  - 0: clock is low when idle
  - 1: clock is high when idle
- Clock phase (CPHA) refers to when data is sampled vs. when new data is shifted out
  - 0: the first clock transition is the first data capture edge
  - 1: the second clock transition is the first data capture edge
- The clock transition (rising or falling) depends on the clock polarity
- 4 combinations or modes (CPOL,CPHA) = (0,0), (0,1), (1,0), (1,1)
- Must pay attention to match this mode to the peripheral!

### Basic Configuration in Master/Controller Mode

- Configure clock tree
- Turn on SPI clock domain
- Set SPI parameters
  - Clock rate using baud rate divisor
  - CPOL and CPHA to match slave
  - DFF to 8- or 16-bit data frame format
  - Set LSBFIRST bit to set whether lsb or msb is sent first (normally msb)
  - Configure the NSS pin (can either use software management or a separate GPIO set as an output and manually toggle it)
  - Set to master mode MSTR
- Enable SPI Set SPE bit to 1

## SPI Demo

### Lab

A Google Sheet for scheduling lab checkoffs can be found here. (g.hmc credentials required)

Any code for the labs may be found on the E155 course Github repository.

- Lab 1 Board Assembly and Testing (PDF)
- Lab 2 Multiplexed 7-Segment Display (PDF)
- Lab 3 ARM Assembly Sort (PDF)
- Lab 4 Keypad Scanner (PDF)
- Lab 5 Digital Audio (PDF)
- Lab 6 The Internet of Things and Serial Peripheral Interface (PDF)
  - DS1722 Datasheet
  - Serial Protocol Decoding on the RIGOL MSO1104z

## spi\_demo.c:main()

```
int main(void) {
     configureFlash();
     configureClock();
     gpioEnable(GPIO_PORT_A);
     gpioEnable(GPIO_PORT_B);
     gpioEnable(GPI0_PORT_C);
     RCC->APB2ENR |= (RCC_APB2ENR_TIM15EN);
10
     initTIM(TIM15);
11
12
     initSPI(2, 0, 0);
13
14
     while(1) {
15
       digitalWrite(PA11, PI0_HIGH);
       spiSendReceive(0xAB);
16
17
       digitalWrite(PA11, PI0_LOW);
18
       delay_millis(TIM15, 10);
19
20 }
```

## **SPI.** h Function Prototypes

```
/* Enables the SPI peripheral and intializes its clock speed (baud rate), polarity, and phase.
2 * -- br[2:0]: (0x0 to 0x7). The SPI clk will be the master clock / clkdivide.
3 * -- cpol: clock polarity (0: inactive state is logical 0, 1: inactive state is logical 1).
4 * -- cpha: clock phase (0: the first clock transition is the first data capture edge,
5 * 1: the second clock transition is the first data capture edge)
6 * Refer to the datasheet for more low-level details. */
void spiInit(int br, int cpol, int cpha);
8

/* Transmits a character (1 byte) over SPI and returns the received character.
10 * -- send: the character to send over SPI
11 * -- return: the character received over SPI */
12 char spiSendReceive(char send);
```

## **An Example SPI Transaction**



## **DS1722 SPI Temperature Sensor**



## DS1722 Digital Thermometer with SPI/3-Wire Interface

www.maxim-ic.com

### **PIN ASSIGNMENT**



### **DETAILED PIN DESCRIPTION** Table 2

| SOIC  | SYMBOL       | DESCRIPTION                                                                                                                                                                                                                                                              |  |  |
|-------|--------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|--|
| PIN 1 | $V_{DDD}$    | <b>Digital Supply Voltage</b> 1.8V-5.5V. Defines the top rails for the digital inputs and outputs.                                                                                                                                                                       |  |  |
| PIN 2 | CE           | <b>Chip Enable</b> Must be asserted high for communication to take place for either the SPI or 3-wire interface.                                                                                                                                                         |  |  |
| PIN 3 | SCLK         | <b>Serial Clock Input</b> Used to synchronize data movement on the serial interface for either the SPI or 3-wire interface.                                                                                                                                              |  |  |
| PIN 4 | GND          | Ground pin.                                                                                                                                                                                                                                                              |  |  |
| PIN 5 | SDO          | <b>Serial Data Output</b> When SPI communication is selected, the SDO pin is the serial data output for the SPI bus. When 3-wire communication is selected, this pin must be tied to the SDI pin (the SDI and SDO pins function as a single I/O pin when tied together.) |  |  |
| PIN 6 | SDI          | <b>Serial Data Input</b> When SPI communication is selected, the SDI pin is the serial data input for the SPI bus. When 3-wire communication is selected, this pin must be tied to the SDO pin (the SDI and SDO pins function as a single I/O pin when tied together.)   |  |  |
| PIN 7 | SERMODE      | Serial Interface Mode Input This pin selects which interface standard will be used: SPI when connected to $V_{\rm CC}$ ; standard 3-wire when connected to GND.                                                                                                          |  |  |
| PIN 8 | $V_{ m DDA}$ | Analog Supply Voltage 2.65V – 5.5V input power pin.                                                                                                                                                                                                                      |  |  |

## **DS1722 Functional Block Diagram**

### **DS1722 FUNCTIONAL BLOCK DIAGRAM** Figure 1



## **Temperature Data Register Format**

### Temperature/Data Relationships Table 3

| Address  |  |  |  |  |
|----------|--|--|--|--|
| Location |  |  |  |  |
| 02h      |  |  |  |  |

|                 |                           |                 |                 |       |       |    |                | Location |
|-----------------|---------------------------|-----------------|-----------------|-------|-------|----|----------------|----------|
| S               | 26                        | 2 <sup>5</sup>  | 24              | $2^3$ | $2^2$ | 21 | 2 <sup>0</sup> | 02h      |
| MSb             | Sb $(unit = {}^{\circ}C)$ |                 |                 |       | LSb   |    |                |          |
| 2 <sup>-1</sup> | 2 <sup>-2</sup>           | 2 <sup>-3</sup> | 2 <sup>-4</sup> | 0     | 0     | 0  | 0              | 01h      |

| TEMPERATURE | DIGITAL OUTPUT<br>(BINARY) | DIGITAL OUTPUT<br>(HEX) |
|-------------|----------------------------|-------------------------|
| +120°C      | 0111 1000 0000 0000        | 7800h                   |
| +25.0625°C  | 0001 1001 0001 0000        | 1910h                   |
| +10.125°C   | 0000 1010 0010 0000        | 0A20h                   |
| +0.5°C      | 0000 0000 1000 0000        | 0080h                   |
| 0°C         | 0000 0000 0000 0000        | 0000h                   |
| -0.5°C      | 1111 1111 1000 0000        | FF80h                   |
| -10.125°C   | 1111 0101 1110 0000        | F5E0h                   |
| -25.0625°C  | 1110 0110 1111 0000        | E6F0h                   |
| -55°C       | 1100 1001 0000 0000        | C900h                   |

### **SPI Transactions**

### Register Address Structure Table 4

| Read Address | Write Address | Active Register |
|--------------|---------------|-----------------|
| 00h          | 80h           | Configuration   |
| 01h          | No access     | Temperature LSB |
| 02h          | No access     | Temperature MSB |

### SPI SINGLE BYTE WRITE Figure 4



### **SPI SINGLE-BYTE READ** Figure 5



# Common Microcontroller Software Interface Standard (CMSIS)

## **CMSIS Major Components**



### The benefits of CMSIS

- Software reusability
- Software compatibility
- Easy to learn and use
- Toolchain independent
- Openness

### **CMSIS-Core Structure**



CMSIS-Core structure

p. 51 The Definitive guide to ARM Cortex-M3 and Cortex-M4 Processors

## **Using CMSIS-Core**



FIGURE 2.14

Using CMSIS-Core in a project

p. 52 The Definitive guide to ARM Cortex-M3 and Cortex-M4 Processors

## Files to Include in a project

- Startup Code (typically startup\_<device>.c/.s)
- Application code (main c)
- <device>.h
  - core\_cm4.h
  - system\_<device>.h
  - system\_<device>.c
- Peripheral Driver files (custom drivers you write or import)
  - For example, the drivers you are writing for lab.

### **Using CMSIS-Core: Startup Files**



FIGURE 2.14
Using CMSIS-Core in a project

p. 52 The Definitive guide to ARM Cortex-M3 and Cortex-M4 Processors

## startup\_<device>.c/.s

STM32L4xx\_Startup.s - Set the initial SP - Set the initial PC == Reset\_Handler, - Set the vector table entries with the exceptions ISR address - Branches to main in the C library (which eventually calls main()).

## stm32l432xx\_Vectors.s

```
_vectors:
           // Internal exceptions and interrupts
           VECTOR __stack_end__
           VECTOR Reset_Handler
           EXC_HANDLER NMI_Handler
           VECTOR HardFault_Handler
 9
           ISR_RESERVED
10
           ISR RESERVED
11
           ISR_RESERVED
12
           ISR_RESERVED
13
           ISR_RESERVED
14
           ISR_RESERVED
15
           ISR_RESERVED
16
           EXC_HANDLER SVC_Handler
17
           ISR_RESERVED
           ISR_RESERVED
18
           EXC_HANDLER PendSV_Handler
19
20
           EXC_HANDLER SysTick_Handler
    . . .
```

Table 46. STM32L41xxx/42xxx/43xxx/44xxx/45xxx/46xxx vector table

| Position | Priority | Type of priority | Acronym                                                                                 | Description                                                                                    | Address                      |
|----------|----------|------------------|-----------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------|------------------------------|
| -        | -        | -                | -                                                                                       | Reserved                                                                                       | 0x0000 0000                  |
| -        | -3       | fixed            | Reset                                                                                   | Reset                                                                                          | 0x0000 0004                  |
| -        | -2       | fixed            | имі                                                                                     | Non maskable interrupt. The RCC Clock<br>Security System (CSS) is linked to the NMI<br>vector. | 0x0000 0008                  |
| -        | -1       | fixed            | HardFault                                                                               | All classes of fault                                                                           | 0x0000 000C                  |
| -        | 0        | settable         | MemManage                                                                               | Memory management                                                                              | 0x0000 0010                  |
| -        | 1        | settable         | BusFault                                                                                | Pre-fetch fault, memory access fault                                                           | 0x0000 0014                  |
| -        | 2        | settable         | UsageFault                                                                              | Undefined instruction or illegal state                                                         | 0x0000 0018                  |
| -        | -        | -                | -                                                                                       | Reserved                                                                                       | 0x0000 001C -<br>0x0000 0028 |
| -        | 3        | settable         | SVCall                                                                                  | System service call via SWI instruction                                                        | 0x0000 002C                  |
| -        | 4        | settable         | Debug                                                                                   | Monitor                                                                                        | 0x0000 0030                  |
| -        | -        | -                | -                                                                                       | Reserved                                                                                       | 0x0000 0034                  |
| -        | 5        | settable         | PendSV                                                                                  | Pendable request for system service                                                            | 0x0000 0038                  |
| -        | 6        | settable         | SysTick                                                                                 | System tick timer                                                                              | 0x0000 003C                  |
| 0        | 7        | settable         | WWDG                                                                                    | Window Watchdog interrupt                                                                      | 0x0000 0040                  |
| 1        | 8        | settable         | PVD_PVM PVD_PVM2 <sup>(1)</sup> /PVM3/PVM4 through EXTI lines 16/35/36/37/38 interrupts |                                                                                                | 0x0000 0044                  |
| 2        | 9        | settable         | RTC_TAMP_STAMP<br>/CSS_LSE                                                              | RTC Tamper or TimeStamp /CSS on LSE through EXTI line 19 interrupts                            | 0x0000 0048                  |
| 3        | 10       | settable         | RTC_WKUP                                                                                | RTC Wakeup timer through EXTI line 20 interrupt                                                | 0x0000 004C                  |
| 4        | 11       | settable         | FLASH                                                                                   | Flash global interrupt                                                                         | 0x0000 0050                  |
| 5        | 12       | settable         | RCC                                                                                     | RCC global interrupt                                                                           | 0x0000 005C                  |
| 6        | 13       | settable         | EXTI0                                                                                   | EXTI Line0 interrupt                                                                           | 0x0000 005C                  |
| 7        | 14       | settable         | EXTI1                                                                                   | EXTI Line1 interrupt                                                                           | 0x0000 005C                  |
| 8        | 15       | settable         | EXTI2                                                                                   | EXTI Line2 interrupt                                                                           | 0x0000 0060                  |
| 9        | 16       | settable         | EXTI3                                                                                   | EXTI Line3 interrupt                                                                           | 0x0000 0064                  |
| 10       | 17       | settable         | EXTI4                                                                                   | EXTI Line4 interrupt                                                                           | 0x0000 0068                  |

### **Using CMSIS-Core: Device Files**



FIGURE 2.14

Using CMSIS-Core in a project

p. 52 The Definitive guide to ARM Cortex-M3 and Cortex-M4 Processors

### <device>.h

- Data structures and the address mapping for all peripherals
- Peripherals registers declarations and bits definition
- Macros to access peripheral's registers hardware

### stm321432xx.h

### Base Addresses

```
** @addtogroup Peripheral_memory_map
     * @{
     */
 4 #define FLASH BASE
                                 (0x08000000UL) /*! < FLASH(up to 256 KB) base address */
 5 #define FLASH END
                                  (0x0803FFFFUL) /*!< FLASH END address
 6 #define FLASH BANK1 END
                                  (0x0803FFFFUL) /*!< FLASH END address of bank1
 7 #define SRAM1 BASE
                                  (0x20000000UL) /*!< SRAM1(up to 48 KB) base address */
 8 #define SRAM2 BASE
                                  (0x10000000UL) /*! < SRAM2(16 KB) base address */
 9 #define PERIPH BASE
                                 (0x4000000UL) /*!< Peripheral base address */
10 #define QSPI_BASE
                                  (0x9000000UL) /*!< QUADSPI memories accessible over AHB base address */
11
12 #define QSPI_R_BASE
                                 (0xA0001000UL) /*!< QUADSPI control registers base address */
13 #define SRAM1 BB BASE
                                 (0x22000000UL) /*!< SRAM1(96 KB) base address in the bit-band region */
14 #define PERIPH BB BASE
                                 (0x42000000UL) /*!< Peripheral base address in the bit-band region */
```

### stm32l432xx.h

### **SPI Register Mapping**

```
1 /**
    * @brief Serial Peripheral Interface
     */
   typedef struct
6
     __IO uint32_t CR1;
                               /*!< SPI Control register 1,
                                                                                         Address offset: 0
    __IO uint32_t CR2;
                              /*!< SPI Control register 2,
                                                                                         Address offset: 0
     __IO uint32_t SR;
                                                                                         Address offset: 0
                               /*!< SPI Status register,
10
    __IO uint32_t DR;
                              /*!< SPI data register,
                                                                                         Address offset: 0
                            /*!< SPI CRC polynomial register,</pre>
11
    __IO uint32_t CRCPR;
                                                                                         Address offset: 0
                            /*!< SPI Rx CRC register,</pre>
12
    __IO uint32_t RXCRCR;
                                                                                         Address offset: 0
    __IO uint32_t TXCRCR;
                              /*!< SPI Tx CRC register,
                                                                                         Address offset: 0
13
14 } SPI_TypeDef;
```

### Bit definitions

```
1 #define GPIO_MODER_MODEO_Pos
                                          (0U)
 2 #define GPIO_MODER_MODE0_Msk
                                          (0x3UL << GPIO MODER MODE0 Pos)
                                                                                  /*!< 0x00000003 */
 3 #define GPIO_MODER_MODE0
                                          GPIO_MODER_MODEO_Msk
 4 #define GPIO MODER MODE0 0
                                          (0x1UL << GPIO MODER MODE0 Pos)
                                                                                  /*!< 0x00000001 */
 5 #define GPIO MODER MODE0 1
                                          (0x2UL << GPIO MODER MODE0 Pos)
                                                                                  /*!< 0x00000002 */
1 /**
     * @brief General Purpose I/O
     */
   typedef struct
 6
     __IO uint32_t MODER;
                              /*!< GPIO port mode register,
                                                                            Address offset: 0x00
                                                                                                      */
     __IO uint32_t OTYPER;
                              /*!< GPIO port output type register,
                                                                            Address offset: 0x04
                                                                                                      */
     IO uint32 t OSPEEDR;
                               /*!< GPIO port output speed register,
                                                                            Address offset: 0x08
                                                                                                      */
     __IO uint32_t PUPDR;
10
                               /*!< GPIO port pull-up/pull-down register,</pre>
                                                                            Address offset: 0x0C
                                                                                                      */
                                /*!< GPIO port input data register,</pre>
11
     IO uint32 t IDR;
                                                                            Address offset: 0x10
                                                                                                      */
12
     __IO uint32_t ODR;
                                /*!< GPIO port output data register.
                                                                            Address offset: 0x14
                                                                                                      */
                               /*!< GPIO port bit set/reset register,
13
     IO uint32 t BSRR;
                                                                            Address offset: 0x18
                                                                                                      */
14
     __IO uint32_t LCKR;
                              /*!< GPIO port configuration lock register, Address offset: 0x1C
                                                                                                      */
     I0 uint32 t AFR[2];
                            /*!< GPIO alternate function registers,</pre>
15
                                                                           Address offset: 0x20-0x24 */
16
     __IO uint32_t BRR;
                                /*!< GPIO Bit Reset register,
                                                                           Address offset: 0x28
                                                                                                      */
17
18 } GPIO_TypeDef;
1 #define GPIOA ((GPIO TypeDef *) GPIOA BASE)
```

### Using bit definitions

### Example code to set PA0 to OUTPUT $(0 \times 1 = 0 \times 0.1)$

```
1 GPIOA->MODER &= ~(0b11 << GPIO_MODER_MODE0_Pos) // Clear bits
2 GPIOA->MODER |= (0b01 << GPIO_MODER_MODE0_Pos) // Set bit 0</pre>
```

## Macros in C (#define)

### Object-like Macros

```
1 #define <TOKEN_NAME> <TOKEN_VALUE>
2 #define BUFFER_SIZE 2056
3 foo = (char *) malloc (BUFFER_SIZE);
```

### Function-like Macros

```
1 #define <MACRO_NAME>(<param1>,<param2>,...) (<stuff to do>)
2
3 #define min(X, Y) ((X) < (Y) ? (X) : (Y))
4
5 x = min(a, b); → x = ((a) < (b) ? (a) : (b));
6 y = min(1, 2); → y = ((1) < (2) ? (1) : (2));</pre>
```

https://gcc.gnu.org/onlinedocs/cpp/Macros.html#Macros

### VAL2FLD Macro

```
1 /**
  \brief Mask and shift a bit field value for use in a register bit range.
  \param[in] field Name of the register bit field.
  \param[in] value Value of the bit field. This parameter is interpreted as an uint32 t type.
                   Masked and shifted value.
  \return
6 */
7 #define _VAL2FLD(field, value) (((uint32_t)(value) << field ## _Pos) & field ## _Msk)
2 #define GPI0_MODER_MODE3_Pos
                                    (6U)
3 #define GPIO MODER MODE3 Msk
                                    (0x3UL << GPIO MODER MODE3 Pos)
                                                                     /*!< 0x000000C0 */
4 #define GPIO_MODER_MODE3
                                   GPIO_MODER_MODE3_Msk
5 #define GPIO MODER MODE3 0
                                   (0x1UL << GPIO MODER MODE3 Pos) /*!< 0x00000040 */
6 #define GPIO_MODER_MODE3_1
                                    (0x2UL << GPIO_MODER_MODE3_Pos)
                                                                      /*!< 0x00000080 */
  0b0000000000000000000000000000000011
3 0b00000000000000000000000000000011 << 6U = 0b0000000000000000000000011000000
```

### \_VAL2FLD Macro

```
1 /**
2 \brief Mask and shift a bit field value for use in a register bit range.
3 \param[in] field Name of the register bit field.
4 \param[in] value Value of the bit field. This parameter is interpreted as an uint32_t type.
5 \return Masked and shifted value.
6 */
7 #define _VAL2FLD(field, value) (((uint32_t)(value) << field ## _Pos) & field ## _Msk)</pre>
```

### Example: Set MODER3 to 0b01 (output)

```
1 _VAL2FLD(GPI0_MODER_MODE3, 0b01)
```

### Expands to

### \_FLD2VAL Macro

### Similar idea to \_VAL2FLD but going the other way

## system\_stm32f4xx.c

- SystemInit(): This function is called at startup just after reset and before branch to main program. This call is made inside the startup\_stm32f4xx.s file.
- SystemCoreClock variable: Contains the core clock (HCLK), it can be used by the user application to setup the SysTick timer or configure other parameters.
- SystemCoreClockUpdate(): Updates the variable SystemCoreClock and must be called whenever the core clock is changed during program execution.