# Microprocessor Systems Lab 3

Checkoff and Grade Sheet

| Partner 1 Name: |  |  |  |  |
|-----------------|--|--|--|--|
|                 |  |  |  |  |
| Partner 2 Name: |  |  |  |  |

| Grade Component                  | Max. | Points Awarded | TA Init.s | Date |
|----------------------------------|------|----------------|-----------|------|
| Performance Verification: Task 1 | 5 %  |                |           |      |
| Task 2                           | 10 % |                |           |      |
| Task 3                           | 5 %  |                |           |      |
| Task 4                           | 20 % |                |           |      |
| Task 5 [Depth]                   | 10 % |                |           |      |

### $\rightarrow$ Laboratory Goals

By completing this laboratory assignment, you will learn to:

- 1. Communicate with devices using UART,
- 2. Communicate with devices using SPI,
- 3. Implement communication using either Polling (Blocking) or Interrupt (Non-Blocking) configurations.

### $\rightarrow$ Reading and References

- R1. Mastering STM32: Chapters 8 (UART), 15 (SPI)
- R2. stm32f769ni\_Datasheet.pdf: Alternate Functions, Table 13
- R3. UM1905-stm32f7\_HAL\_and\_LL\_Drivers.pdf: Chapters 62 (SPI), 66 (UART)
- R4. RM0410-stm32f7\_Reference\_Manual.pdf: Chapters 34 (USART), 35 (SPI/I2S)
- R5. Lab03\_UART\_SPI\_Template.zip: Project Template for Lab 3
- R6. STaTS\_Device\_Datasheet.pdf: Example datasheet for peripheral device incorporating an SPI interface. Flashing and use instructions are near end of document.
- R7. LAB-03\_STaTS\_Firmware-F303RE.bin: Firmware for example SPI device, Nucleo-F303RE.
- R8. LAB-03\_STaTS\_Firmware-F303K8.bin: Firmware for example SPI device, Nucleo-F303K8.

### **Serial Communication**

In order to transport data into or out off a microcontroller from other digital devices (e.g., sensors, computers, other microcontrollers), various communication standards may be used, such as I<sup>2</sup>C<sup>1</sup>, UART, SPI, CAN, etc. These listed standards are all examples of *Serial Communication* techniques, where data is transferred bit-by-bit. Other standards exist but are not common in microcontroller applications, such as PATA and SCSI along with other various memory access methods, are known as *Parallel Communication*, where byte(s) are transferred over many communication lines at once<sup>2</sup>. For this lab, we will focus on the UART (Universal Asynchronous Receiver Transmitter) and SPI (Serial Peripheral Interface) standards.

Within the serial communication domain, there are two distinct communication methods: synchronous and asynchronous. These are differentiated by whether the data clock is shared between the devices in communication or not. In an asynchronous communication system (e.g., UART), a clock is not shared between the devices; requiring that all devices on the communication bus have knowledge of the expected data baud rate<sup>3</sup> For example: all labs in this course use terminal communication between the STM32F769NI and the computer which is achieved through a UART interface. When configuring the terminal on the computer, the baud rate is explicitly specified to the terminal program as 115200 bps.

<sup>&</sup>lt;sup>1</sup>The I3C standard has been released towards replacing I<sup>2</sup>C, though adaption may take a while

<sup>&</sup>lt;sup>2</sup>The STM32F769NI has a Flexible Memory Controller (FMC) which contains a parallel interface for memory addressing up to 32-bits wide.

<sup>&</sup>lt;sup>3</sup>Baud rate for our purposes is the communication *bit rate* or *data rate*, measured in bits per second. In higher-order modulated communication systems, baud rate is also known as the *symbol rate*.

Alternatively, synchronous communications are those where the data clock is shared between the devices on the bus (e.g., I<sup>2</sup>C, SPI). This allows for the devices receiving the clock to not require an onboard timebase generator but instead requires all devices to have an extra I/O line to receive/emit the clock signal, as well as requiring an additional line connected each device. Sharing the clock between the devices also increases the maximum throughput of the bus, as there is less required overhead for data packet control signals. Synchronous communication techniques commonly have a Controller/Peripheral architecture<sup>4</sup>, where one device on the bus is configured as the controller and directly controls the communication bus<sup>5</sup>, which includes generation and sharing of the data clock. The peripherals only communicate when addressed (I<sup>2</sup>C) or selected (SPI) by the controller.

# Universal Asynchronous Receiver Transmitter (UART)

The STM32F769NI has several on-board universal synchronous/asynchronous receiver/transmitter (US-ART), which may be used to generate UART, among other standards. The DISCO board has one built-in virtual UART communication channel over USB which USART1 is configured to use. Each UART interface requires only two signal lines: RX (receive data) and TX (transmit data). These lines are named the same for any device connected, therefore, for device 1 to transmit data to device 2, the TX line of device 1 must be connected to the RX line of device 2 and vice versa. The USARTs can of course be configured through the registers listed in R4, though for this lab, the hardware abstraction layer (HAL) drivers will be used. For each USART configured to operate as a UART device, a UART\_HandleTypeDef type module handle is required, which has the following defined structure:

The UART\_InitTypeDef Init field is another struct containing the following parameters, which set up how the UART operates to exchange data:

```
typedef struct {
    uint32_t BaudRate;
    uint32_t WordLength;
    uint32_t StopBits;
    uint32_t Parity;
    uint32_t Mode;
    uint32_t HwFlowCtl;
    uint32_t OverSampling;
} UART_InitTypeDef;
```

These types of module configuration handles that are reused throughout the code (this includes: UART\_HandleTypeDef, SPI\_HandleTypeDef, etc.) should always be defined as global variables.

<sup>&</sup>lt;sup>4</sup>The traditional terminology of "controller/peripheral" was "master/slave". Recently, multiple naming conventions have been arising such as "primary/secondary," "leader/follower," or Python's recent change to "parent/worker."

<sup>&</sup>lt;sup>5</sup>There are multi-controller techniques, we will only focus on single-controller SPI.

### $\rightarrow$ UART Port Setup

In the basic mode, the UART is relatively easy to configure when using the HAL. Three general steps are required:

- 1. Enable GPIO port for transmitting and receiving
- 2. Populating the UART\_HandleTypeDef fields Instance and Init.
- 3. Call HAL\_UART\_Init()

Note that item 3 above does not and cannot inherently accomplish item 1 as there are many different configurations as to which GPIO pins to place the UART signals. While this alternatively may have been accomplished through another field in the UART\_HandleTypeDef (e.g., GPIOInit), the authors of the HAL instead have the HAL\_UART\_Init() function call back to the user space function HAL\_UART\_MspInit(), where all GPIO configuration is expected to be done. The callback function HAL\_UART\_MspInit() is provided in the template project R5 in file uart.c.

Transmitting and receiving data is also a simple procedure requiring only HAL\_UART\_Transmit() and HAL\_UART\_Receive(). The functionality of printf(), putchar(), and getchar() are all implemented using those two functions (again, see R5 uart.c).

# ♦ Task 1: Two-Terminal UART (Polling)

Write a program that will monitor two serial ports continuously. The STM32F769NI has 8 total UART capable modules, though only 5 are accessible on the DISCO board, one of these being USART1 over USB. The other port used should be USART6. There are three options for development here:

- 1. The USART6 interfaces between each group member's DISCO board may be connected together,
- 2. The "STaTS" device R6 may be used as the second unit<sup>6</sup>, or
- 3. USART6 may be connected to a computer using a USB-to-serial adapter, available in the classroom.

USART1 should be left as configured in the template project and USART6 should be configured to use 38400 baud and N-8-1 (no parity bits, 8 data bits, 1 stop bit). For reference, USART1 is configured as 115200 baud N-8-1.

Whenever the program detects a character coming in from either of the onboard serial ports, it should echo that character back to both serial ports. This requires two terminal programs to be running; though they do not necessarily have to be on the same computer. When <ESC> is pressed on either terminal, a brief exit message on both screens should be shown and the program halted. Since continuous polling of both ports is required, getchar() may not be used as it will wait indefinitely for a key press.

 $<sup>^6</sup>$ Instructions for operating the DISCO board and Nucleo board simultaneously on the same computer are listed on the last page of this document

#### NOTES:

- 1. If using an USB-to-serial adapter that complies with RS-232, the TX and RX lines will be digitally inverted. This must be fixed.
- 2. Ensure the terminals are configured with the correct baud rate: 115,200 and 38,400.
- 3. Failure to connect the Grounds between the DISCO and second device will result in either no output or garbage output.
- 4. Remember that TX1 should be connected to RX2, TX2-RX1.
- 5. Much of what you need already exists in uart.c and init.c.
- 6. The timeout values for HAL\_UART\_Transmit() and HAL\_UART\_Receive() may be very small.
- 7. Be careful when using uart\_getchar() for this task. The function will always return a value when HAL\_UART\_Receive() timeout, and the value might not be what is expected.

## Polling Versus Interrupt Operation

In Task 1, a program was made that continuously checked whether any characters were received on USART1 or USART6. This type of implementation is known as *polling*, or continuously (or periodically) checking to see if a specific event has happened. This method has the drawback that to successfully detect an event in complex code at the proper time, checks of the event need to be placed throughout the code. This both makes the code mode difficult to write and maintain but also introduces computational overhead. Alternatively, if only one check for the event is placed within the code and will not continue until the event has happened, then the program is essentially "blocked" from doing anything else.

This functionality can instead be handled by interrupts, or in a "non-blocking" mode. This frees up the additional overhead of polling while also ensuring that the event is dealt with immediately, in cases where there is a time sensitivity (e.g., Real Time systems). For the case of UART character reception, it is desired to trigger interrupts when a character is received by either UART port. To implement the IRQHandler for USART1, the following code should be used:

where USB\_UART is the handle for USART1, defined in the template project. A similar function will need to be written for USART6. Additionally, the callback function HAL\_UART\_RxCpltCallback() will need to be written to handle character reception. Note that the handle of the USART module that triggered the interrupt will be passed to this callback function and would be used to determine the character's origin. In order receive in this fashion, HAL\_UART\_Receive\_IT() will need to be used instead of HAL\_UART\_Receive().

# ♦ Task 2: Two-Terminal UART (Interrupt)

Reimplement task 1 except using a non-blocking interrupt structure instead of the polling method. The while loop in the main function **cannot** have any UART commands contained within it.

#### NOTES:

- 1. The debugging mode of the IDE may be very useful for this task in order to determine if a character input is being handled properly.
- 2. Don't forget to use the NVIC to enable the interrupts.
- 3. HAL\_UART\_RxCpltCallback() will only be called once per one call of HAL\_UART\_Receive\_IT(). The receive another character, this command would have to be reissued somehow.
- 4. While UART commands are not allowed within the main program loop, they are allowed prior to the loop; initialization commands as well as transmit and receive commands.

# Serial Peripheral Interface (SPI)

Note: The traditional names for the devices in SPI and I<sup>2</sup>C communication schemes: **master** and **slave**, have been replaced with **controller** and **peripheral**. The STM32 device documentation maintains the **master/slave** convention. See here for a discussion on this change.

As introduced already, one synchronous protocol typically available on microcontrollers is the SPI. This protocol is essentially an alternative to the I<sup>2</sup>C. There are significant differences between the two, mainly arising from peripheral selection and data lines. Active peripheral selection in I<sup>2</sup>C is done via a software address, where the address of a peripheral is passed first in each data packet to specify which peripheral is supposed to read from or write to the controller. The SPI uses "chip select" (CS, also labelled SS or NSS), one for each peripheral. Employing the chip select eliminates the overhead required in the I<sup>2</sup>C Bus of sending information to select the peripheral, allowing for much higher achievable throughputs; however, this comes at the expense of requiring a CS line for each peripheral on the bus (hardware complexity). Of course, the hardware required to support SPI could become exceedingly burdensome if there are many peripherals on the bus, thereby requiring the use of I<sup>2</sup>C instead. A generic hardware layout for each of these standards is given in Figure 1.

In addition to multiple CS lines for the SPI, the SPI also uses two data lines: Controller In, Peripheral Out (CIPO) and Controller Out, Peripheral In (COPI)<sup>7</sup>; as opposed to I<sup>2</sup>C using only SDA. The use of CIPO and COPI allows for bidirectional communication to occur, both being clocked by SCLK, again, allowing for higher possible throughput. Data transmission and reception are always done concurrently, that is, for every byte sent, a byte is received. However, distinctions are made for SPI transmitting and SPI receiving only modes:

- 1. Transmit Mode: Data is transmitted on COPI while data received on CIPO is ignored.
- 2. Receive Mode: Dummy data is transmitted on COPI (e.g., 0x00 or 0xFF) while data is received on CIPO and stored.

Unlike the I<sup>2</sup>C, SPI is less standardized; which results in many different methods to communicate with a peripheral. The communication requirements of the peripheral will need to be tailored to individually. Two points of interest are the required behavior of the clock, or the **clock polarity**, and when to change/latch (read) data, or the **clock phase**. The clock polarity can either be *Idle High* or *Idle Low*, indicating the state of SCLK when not communicating. The clock phase also has two possibilities: either the first clock edge or second clock edge indicates data capture (rising or falling), with the other edge being the data change signal. Other common differences between the devices is the absence of either the CIPO or COPI lines, or bidirectional communication on only one of the lines, similar to I<sup>2</sup>C.

One aspect that is consistent however is that selection of the peripheral is done via pulling the CS lines low. This is indicated by the notation in Figure 1, where " $^{\sim}$ CS" is shown: the  $^{\sim}$  indicating that the signal is  $Active\ Low$  as opposed to  $Active\ High$ . This implies that when  $^{\sim}$ CS for the peripheral is high, the peripheral will release control of CIPO ignore all activity of the bus. This is also commonly denoted with a slash or overline instead:  $^{\sim}$ CS or  $^{\sim}$ CS. For the SPI modules on the STM32F769NI, only one  $^{\sim}$ CS line exists, labeled NSS ( $Not\ Slave\ Select$ ). Therefore, the module itself is only capable of communicating with one peripheral by default. To add additional peripherals, GPIO pins must be configured independently to act as the required  $^{\sim}$ CS lines.

<sup>&</sup>lt;sup>7</sup>The signals CIPO and COPI may also be named SDO or SDI for "Serial Data Out" or "Serial Data In".



Figure 1: General architectures of (left) I<sup>2</sup>C and (right) SPI Buses.

### $\rightarrow$ SPI Port Setup

The following steps are required to configure and use the SPI port:

- 1. Configure GPIO pins. These must be configured in alternate function mode. See R2 Table 13.
- 2. Configure SPI clock rate, wire mode (full-duplex or single bidirectional line), clock polarity and phase, and controller or peripheral mode. For this lab's purposes, CRC should be disabled.
- 3. Call HAL\_SPI\_Init()

As was the case for the USART modules, the completion of item 1 is triggered by item 3 through the callback function HAL\_SPI\_MspInit(). In order to send and receive data, the functions HAL\_SPI\_Transmit(), HAL\_SPI\_Receive(), or HAL\_SPI\_TransmitReceive() would be used, or their \_IT() or \_DMA() counterparts if operating in interrupt or DMA mode, respectively.

If the peripheral being used is controlled by the module's NSS line, then no extra commands need to be done to select the peripheral. If, however, a GPIO output pin is used to select the peripheral, the pin needs to manually be asserted low prior to calling a transmit/receive function while also disabling the NSS pin. Once the transfer is complete, the GPIO pin should again be raised high.

### ♦ Task 3: SPI Loopback Interface

Write a program that sets up SPI2 in 8-bit mode, monitors the terminal, echos any character received on the terminal to the SPI bus, and writes any received characters from the SPI bus to the terminal. Wire the SPI bus such that the MISO and MOSI signals of one DISCO board only are connected together (i.e., the loopback condition). The SPI port should be configured to operate at roughly 1 MHz. The terminal should be split top and bottom such that characters received from the keyboard are written on the top half and characters received from the SPI are written on the bottom half. It is not necessary to scroll both the top half and the bottom half, though the exact implementation is up to you. Implementation of this interface either in a Polling or Interrupt Mode is acceptable. The GPIO pins required to implement the SPI bus must be enabled in Alternate Function mode; where the correct mode must be identified. Tables identifying the alternate functions for each pin may be found in ?? Table 13. See uart.c for how this was done for the UART buses.

#### NOTES:

- 1. For this task, clock polarity and phase are not important.
- 2. Although SCLK is not required for this implementation, the SPI bus will not operate if the GPIO line is not properly configured.
- 3. Additionally, the NSS needs to be configured for the SPI to work; leaving the configuration blank, or filling it will the wrong value, will prevent the SPI from operating.
- 4. Test your interface by removing the loopback condition and tying MISO to low or high. This should result in returned values of 0x00 or 0xFF, respectively.

### ♦ Task 4: Connect to STM32-Based SPI Device

Using the datasheet provided in R6, design and implement an SPI controller that can interact with the "STaTS" device and perform the following:

- 1. Reliably send terminal characters from the DISCO to the peripheral device using SPI.
- 2. Receive terminal characters from the peripheral device using SPI. The received characters should be printed on the bottom half of the controller's terminal.
- 3. Read the peripheral's firmware version upon startup.
- 4. Trigger a temperature measurement and retrieve the result when it is ready. The temperature should be printed on the right side of the terminal to avoid the transmitted and received terminal characters.
- 5. Clear or reset the peripheral terminal.
- 6. Change and read the device ID of the peripheral.
- 7. Pressing ESC in the controller's terminal should present a menu where the user can trigger items 2-6. When this menu is active, it is acceptable to ignore characters inputted from the peripheral's terminal.

Sending and receiving terminal characters should happen concurrently (at the same time) and be considered the base-state of the program; that is, when the program isn't performing another function (e.g., changing the device ID), character transmission and reception should be active by default.

You must verify the SPI bus operation using an oscilloscope (or Analog Discovery board, etc.) and include in your report. Using a benchtop oscilliscope to debug is *highly encouraged*. If using an Analog Discovery board, etc. it is best to use the Digital logic lines to watch all five (SCLK,CIPO,COPI,CS,DEBUG) lines at once and trigger on CS falling edges.

#### NOTES:

- Sub-tasks 2 and 6 may be omitted for a satisfactory checkoff; however, they are each worth 4 points towards this task (the task is worth 20 points total).
- The SPI peripheral for this task is a Nucleo-F303RE board. The firmware provided in R8 needs to be programmed onto the Nucleo board as described in R6.
- Triggering of the temperature measurement may be done by the terminal menu or through a timing interval. If done with a timing interval, the menu option for this may be omitted.

- A warning: the temperature measurement will not make sense as it is an internal core temperature which will likely be much higher than room temperature.
- The SPI peripheral may contain some bugs. Alert the instructor if the peripheral is not behaving as described.

# ♦ Task 5: [Depth] Serial "Bit Banging"

In some applications, peripheral support for certain serial communication buses may not be available. The bus must be implemented in code in these situations, as opposed to using specialized hardware. This method is known as "bit banging". Bit banging effectively requires the efficient use of GPIO, timers, and interrupts to produce the bus.

For this task, generate an SPI-like bus transmit path; where the device generates a serial clock, **TX-CLK**, and a serial data output, **SDO**. The clock should only be active when transmitting bits. Similarly, provide a receive transmit path; where the device receives a serial clock, RXCLK, and a serial data input, **SDI**. This topology is describes in Figure 2 below.



Figure 2: Topology of custom serial bus.

To demonstrate the bus, simply pass terminal characters in both directions: terminal input on Device 1 appears on Device 2's terminal and vice versa.

#### NOTES:

- Assume that 8 bits are transferred at a time and the devices stay perfectly synchronized (e.g., no CS lines are required to provide selection or synchronization).
- The bus should be capable of simultaneously transmitting and receiving. This implies that the individual path implementations are non-blocking.
- All other bus specifications are up to the programmers; such as: data rate, clock phase/polarity, etc.
- Microcontroller startup activity (e.g., GPIO initialization) may produce something that looks like a clock edge. Instead of trying to remove this, it is suggested that the receiver doesn't start listening for transmissions until after a startup delay (e.g., 1-2 seconds). If both devices are reset/started at roughly the same time, then these start-up glitches will not be impactful.

# Operating DISCO and Nucleo Boards on Same Computer

In order to complete most tasks of this lab, both boards will likely need to be connected to the same computer simultaneously unless using a second computer. Connection of the PuTTY, or other terminals, is fairly straight forward:

- Windows: The virtual COM port presented by each board on Windows generally stays consistent when connecting to the same USB port; therefore, tracking and determination of these ports is clear.
- Linux: The /dev/ttyACM\# ports are assigned by first-come/first-serve. Therefore, the first device attached will be assigned to /dev/ttyACMO and the next will be /dev/ttyACM1.
- MacOS: I am unsure of MacOS's operation here good luck!

The connection to STM32CubeIDE's programmer and debugger is less clear; however, once configured properly a single device can be tied to a single project. To do so:

- 1. Plug in the DISCO board ONLY.
- 2. Right click the project to associate the DISCO board to.
- 3. Select Properties (Bottom of context menu)  $\rightarrow Run/Debug$  Settings (On Left).
- 4. Select the existing launch configuration and click Edit...
- 5. Select the *Debugger* tab
- 6. Enable the  $ST\text{-}LINK\ S/N$  box and click Scan
- 7. If only the DISCO board is connected, a single serial number will be available within the dropdown box next to ST-LINK SN. Select it and click OK on the bottom.
- 8. This specific DISCO board will now be programmed by the project. If a different DISCO board is used, the  $STLINK\ S/N$  must be adjusted.
- 9. With the configuration set above, the NUCLEO board may be plugged in.

Programming of binary firmware to the Nucleo Board should be done without the DISCO board connected to avoid connection to the wrong development board.