# ECE 272 Lab 5 Fall 2018

# Serial Peripheral Interface (SPI) Phi Luu

November 14<sup>th</sup>, 2018

Grading TA: Edgar Perez

Lab Partner: Benjamin Geyer

### 1 Introduction

Serial Peripheral Interface (SPI) is a method of transmitting multiple bits of data using minimal hardware. When two pieces of hardware communicates with each other via SPI, they need to share a clock. At every rising edge of the clock, the data is transmitted from one device to the other.



Figure 1: The waveform diagram of the master and the slave in SPI communication protocol [1]

In Figure 1, at every rising edge of SCK (clock from master), the data MOSI (master-out slave-in) is transmitted from the master device to the slave device. After 8 rising edges, an 8-bit data will have been transmitted—in this case, 11001010 from master to slave.

When the data goes over the designated number of bits, in this case 8 bits, the SPI will always "forget" the left most bit and transmit the latest chunk of bits.

In this lab, I and Ben designed a SPI protocol using the DE10-Lite FPGA. We used a button as a clock signal, a switch as the data line, and six seven-segment displays as the indicator of the data transmission from the SPI.

## 2 Design

The system takes in two inputs: one for the clock signal, controlled by a push button, and one for the data, controlled by a switch. This 2-bit input then goes to the SPI. The SPI protocol always outputs an 8-bit binary into a parser to pass into a seven-segment display decoder. Finally, the decoder outputs to the LEDs, indicating the current data being transmitted by the SPI protocol, as illustrated by the block diagram in Figure 2 below.

# DE10-Lite



Figure 2: Block diagram

The SPI module takes in the clock signal and the data. It outputs an 8-bit to the parser, and every time the clock rises, it shifts the current value of the output 1 bit to the left and then add the input bit (data) to the right.

$$output = (output << 1) + data$$
 (1)

For example, let the initial value of the output is 00000000, if the data is 1 when the clock rises, the output then becomes

$$output = (00000000 << 1) + 1 = 00000001 \tag{2}$$

Suppose for the next 7 times the clock rises, the data is 1100110, then output = 11100110.

If the clock rises again, output will be overflow. In this case, the SPI will "forget" the oldest bit (the most significant bit), the first 1. Suppose when the clock rises again, the data is a 1. Then, output = 11001101.

The SPI module has the same design as explained above. We reuse the parser and the decoder modules, thus there is nothing new in these modules. However, for the sake of clarity, I will explain

the functionality of the parser and the decoder below.

The parser takes in an 8-bit binary number, and SystemVerilog automatically converts numbers between binary, hex, and decimal systems. Since the SPI result is always an 8-bit number, its maximum value is a 3-digit decimal 255. Therefore, we only need the first three seven-segment displays. The parser outputs three 4-bit numbers, which are then taken in as inputs for the decoders. The decoder module uses the values in Table 1 to convert these numeric inputs to LED signals.

See the Appendix for the code.

| Decimal | Binary | $\mathbf{Seg}_{\mathbf{A}}$ | $\mathrm{Seg}_{\mathrm{B}}$ | $\mathbf{Seg}_{\mathbf{C}}$ | $\operatorname{Seg}_{\operatorname{D}}$ | $\mathbf{Seg}_{\mathbf{E}}$ | $\mathbf{Seg_F}$ | $\operatorname{Seg}_{\mathbf{G}}$ |
|---------|--------|-----------------------------|-----------------------------|-----------------------------|-----------------------------------------|-----------------------------|------------------|-----------------------------------|
| 0       | 0000   | 0                           | 0                           | 0                           | 0                                       | 0                           | 0                | 1                                 |
| 1       | 0001   | 1                           | 0                           | 0                           | 1                                       | 1                           | 1                | 1                                 |
| 2       | 0010   | 0                           | 0                           | 1                           | 0                                       | 0                           | 1                | 0                                 |
| 3       | 0011   | 0                           | 0                           | 0                           | 0                                       | 1                           | 1                | 0                                 |
| 4       | 0100   | 1                           | 0                           | 0                           | 1                                       | 1                           | 0                | 0                                 |
| 5       | 0101   | 0                           | 1                           | 0                           | 0                                       | 1                           | 0                | 0                                 |
| 6       | 0110   | 0                           | 1                           | 0                           | 0                                       | 0                           | 0                | 0                                 |
| 7       | 0111   | 0                           | 0                           | 0                           | 1                                       | 1                           | 1                | 1                                 |
| 8       | 1000   | 0                           | 0                           | 0                           | 0                                       | 0                           | 0                | 0                                 |
| 9       | 1001   | 0                           | 0                           | 0                           | 0                                       | 1                           | 0                | 0                                 |

Table 1: Conversion table between decimal, 4-bit binary, and seven-segment signals

### 3 Results

We upload the project to the real FPGA. We do thorough tests on the real board, and everything seems to be working as intended. Therefore, we have successfully implemented an SPI transmission protocol using SystemVerilog.

## 4 Experiment Notes

#### Reflection

### **Study Questions**

1. In your own words, describe the functionality of 4 wire SPI.

## **Appendix**

38

39

lab5.qsf (Pin Assignment)

set\_location\_assignment PIN\_C14 -to seg70[0]
set\_location\_assignment PIN\_E15 -to seg70[1]

```
set_location_assignment PIN_C15 -to seg70[2]
    set_location_assignment PIN_C16 -to seg70[3]
41
    set_location_assignment PIN_E16 -to seg70[4]
42
    set_location_assignment PIN_D17 -to seg70[5]
43
    set_location_assignment PIN_C17 -to seg70[6]
44
    set_location_assignment PIN_C18 -to seg71[0]
45
    set_location_assignment PIN_D18 -to seg71[1]
46
    set_location_assignment PIN_E18 -to seg71[2]
47
    set_location_assignment PIN_B16 -to seg71[3]
48
    set_location_assignment PIN_A17 -to seg71[4]
49
    set_location_assignment PIN_A18 -to seg71[5]
50
    set_location_assignment PIN_B17 -to seg71[6]
51
    set_location_assignment PIN_B20 -to seg72[0]
52
    set_location_assignment PIN_A20 -to seg72[1]
53
    set_location_assignment PIN_B19 -to seg72[2]
54
    set_location_assignment PIN_A21 -to seg72[3]
55
    set_location_assignment PIN_B21 -to seg72[4]
56
    set_location_assignment PIN_C22 -to seg72[5]
57
    set_location_assignment PIN_B22 -to seg72[6]
58
    set_location_assignment PIN_F21 -to seg73[0]
59
    set_location_assignment PIN_E22 -to seg73[1]
60
    set_location_assignment PIN_E21 -to seg73[2]
61
    set_location_assignment PIN_C19 -to seg73[3]
    set_location_assignment PIN_C20 -to seg73[4]
63
    set_location_assignment PIN_D19 -to seg73[5]
64
    set_location_assignment PIN_E17 -to seg73[6]
65
    set_location_assignment PIN_F18 -to seg74[0]
66
    set_location_assignment PIN_E20 -to seg74[1]
67
    set_location_assignment PIN_E19 -to seg74[2]
68
    set_location_assignment PIN_J18 -to seg74[3]
69
    set_location_assignment PIN_H19 -to seg74[4]
70
    set_location_assignment PIN_F19 -to seg74[5]
71
    set_location_assignment PIN_F20 -to seg74[6]
72
    set_location_assignment PIN_J20 -to seg75[0]
73
    set_location_assignment PIN_K20 -to seg75[1]
74
    set_location_assignment PIN_L18 -to seg75[2]
75
    set_location_assignment PIN_N18 -to seg75[3]
76
    set_location_assignment PIN_M20 -to seg75[4]
77
    set_location_assignment PIN_N19 -to seg75[5]
78
    set_location_assignment PIN_N20 -to seg75[6]
79
80
81
    set_global_assignment -name FAMILY "MAX 10"
    set_global_assignment -name DEVICE 10M50DAF484C7G
82
    set_global_assignment -name TOP_LEVEL_ENTITY SpiTopLevel
83
    set_global_assignment -name ORIGINAL_QUARTUS_VERSION 18.0.0
84
    set_global_assignment -name PROJECT_CREATION_TIME_DATE "11:04:13 OCTOBER 31, 2018"
85
    set_global_assignment -name LAST_QUARTUS_VERSION "18.0.0 Lite Edition"
```

```
set_global_assignment -name PROJECT_OUTPUT_DIRECTORY output_files
     set_global_assignment -name MIN_CORE_JUNCTION_TEMP 0
88
     set_global_assignment -name MAX_CORE_JUNCTION_TEMP 85
89
     set_global_assignment -name ERROR_CHECK_FREQUENCY_DIVISOR 256
90
     set_global_assignment -name POWER_PRESET_COOLING_SOLUTION "23 MM HEAT SINK WITH 200 LFPM
91
     \hookrightarrow AIRFLOW"
     set_global_assignment -name POWER_BOARD_THERMAL_MODEL "NONE (CONSERVATIVE)"
92
94
     set_location_assignment PIN_C10 -to switch
95
     set_location_assignment PIN_B8 -to button
96
     set_global_assignment -name SYSTEMVERILOG_FILE SpiTopLevel.sv
97
     set_global_assignment -name SYSTEMVERILOG_FILE Spi.sv
98
     set_global_assignment -name SYSTEMVERILOG_FILE Parser.sv
99
     set_global_assignment -name SYSTEMVERILOG_FILE Decoder.sv
100
     set_global_assignment -name PARTITION_NETLIST_TYPE SOURCE -section_id Top
     set_global_assignment -name PARTITION_FITTER_PRESERVATION_LEVEL PLACEMENT_AND_ROUTING
102
     \rightarrow -section_id Top
     set_global_assignment -name PARTITION_COLOR 16764057 -section_id Top
103
     set_instance_assignment -name PARTITION_HIERARCHY root_partition -to | -section_id Top
104
```

#### SpiTopLevel.sv

```
module SpiTopLevel(
1
         input logic switch,
2
         input logic button,
3
         output logic [6:0] seg70, seg71, seg72, seg73, seg74, seg75
4
    );
5
6
         logic [7:0] parser_in;
7
         logic [3:0] parser_out0, parser_out1, parser_out2;
8
9
         Spi s(
10
             .clk(button),
11
              .data_in(switch),
12
              .out_parser(parser_in)
13
         );
15
         Parser p(
16
              .in(parser_in),
17
              .out0(parser_out0),
18
              .out1(parser_out1),
19
              .out2(parser_out2)
20
         );
21
22
```

```
Decoder d0(
              .num_in(parser_out0),
24
              .segments(seg70)
25
         );
26
         Decoder d1(
27
              .num_in(parser_out1),
28
              .segments(seg71)
29
         );
         Decoder d2(
31
              .num_in(parser_out2),
32
              .segments(seg72)
33
         );
34
         Decoder d3(
35
              .num_in(0),
36
              .segments(seg73)
37
         );
         Decoder d4(
39
              .num_in(0),
40
              .segments(seg74)
41
         );
42
         Decoder d5(
43
              .num_in(0),
              .segments(seg75)
         );
46
47
     endmodule
48
```

#### Spi.sv

```
module Spi(
1
         input logic clk,
2
         input logic data_in,
3
         output logic [7:0] out_parser
4
    );
5
6
         always_ff @(posedge clk)
             out_parser <= (out_parser << 1) + data_in;</pre>
8
9
    endmodule
10
```

Parser.sv

```
module Parser(
1
         input logic [7:0] in,
2
         output logic [3:0] out0, out1, out2
3
    );
4
5
         always_comb begin
6
             out0 = in % 10;
             out1 = (in / 10) \% 10;
8
             out2 = (in / 100) % 10;
9
         end
10
11
    endmodule
12
```

#### Decoder.sv

```
module Decoder(
         input logic [3:0] num_in,
2
         output logic [6:0] segments
3
    );
4
         always_ff @(*)
5
             case(num_in)
6
                 0: segments = 7'b100_0_000;
7
                 1: segments = 7'b111_1_001;
                 2: segments = 7'b010_0_100;
9
                 3: segments = 7'b011_0_000;
10
                 4: segments = 7'b001_1_001;
11
                 5: segments = 7'b001_0_010;
12
                 6: segments = 7'b000_0_0_010;
13
                 7: segments = 7'b111_1_000;
                 8: segments = 7'b000_0_0000;
                 9: segments = 7'b001_1_000;
16
                 default: segments = 7'b111_1111;
17
             endcase
18
19
    endmodule
20
```

## References

[1] M. Grusin, "Serial peripheral interface (spi)." https://learn.sparkfun.com/tutorials/serial-peripheral-interface-spi.