



**INDIAN INSTITUTE OF INFORMATION  
TECHNOLOGY, DESIGN AND  
MANUFACTURING**  
**KURNOOL**

**Design and Implementation of SPI  
Protocol**

**Using Two ZedBoards (Master and Slave)**

**Submitted by:**

**V. Abhinav**

**Roll No: 523EC0013**

**Department of Electronics and Communication Engineering**

**Under the Guidance of  
DR P.RANGA BABU SIR**

# Contents

|                                                                    |           |
|--------------------------------------------------------------------|-----------|
| <b>Abstract</b>                                                    | <b>2</b>  |
| <b>1 Introduction</b>                                              | <b>3</b>  |
| <b>2 Objectives</b>                                                | <b>3</b>  |
| <b>3 Tools Used</b>                                                | <b>3</b>  |
| <b>4 Description of SPI Protocol</b>                               | <b>4</b>  |
| 4.1 Working Principle of SPI Communication . . . . .               | 4         |
| 4.1.1 1. Clock Synchronization . . . . .                           | 4         |
| 4.1.2 2. Data Transmission . . . . .                               | 4         |
| 4.1.3 3. Data Reception at the Slave . . . . .                     | 5         |
| 4.1.4 4. Master-Slave Synchronization Logic . . . . .              | 5         |
| 4.1.5 5. Timing Considerations . . . . .                           | 5         |
| 4.2 Connection Diagram . . . . .                                   | 5         |
| <b>5 Verilog HDL Implementation</b>                                | <b>6</b>  |
| 5.1 Master Module Code . . . . .                                   | 6         |
| 5.2 Master Test bench code . . . . .                               | 8         |
| 5.3 Master board top level module code . . . . .                   | 9         |
| 5.4 Master board Constraints (XDC) File . . . . .                  | 10        |
| 5.5 Slave Module Code . . . . .                                    | 11        |
| 5.6 Slave test bench Code . . . . .                                | 13        |
| 5.7 Slave board top level module code . . . . .                    | 14        |
| 5.8 Slave board Constraints(XDC) File . . . . .                    | 15        |
| 5.9 Schematic diagram of Master and Slave Communications . . . . . | 17        |
| <b>6 ZedBoard Implementation</b>                                   | <b>18</b> |
| <b>7 Results and Discussion</b>                                    | <b>19</b> |
| <b>8 Advantages</b>                                                | <b>20</b> |
| <b>9 Conclusion</b>                                                | <b>20</b> |

# Abstract

The Serial Peripheral Interface (SPI) protocol is one of the most efficient synchronous communication systems used for high-speed data transfer between digital devices. This project presents the design and implementation of SPI protocol using two Digilent Zed-Boards, where one acts as a **Master** and the other as a **Slave**. The Master board generates and transmits an 8-bit data pattern, which is received and displayed on LEDs connected to the Slave board. The entire design is realized using **Verilog HDL** on the Xilinx Vivado platform. The project highlights synchronous serial communication principles, FPGA hardware interfacing, and timing control for reliable data transmission.

# 1 Introduction

Serial Peripheral Interface (SPI) is a full-duplex synchronous communication protocol that allows data exchange between one Master and multiple Slave devices. It is widely used in embedded systems due to its simplicity and speed advantages over other serial interfaces such as UART and I<sup>2</sup>C.

In this project, SPI communication is implemented using two ZedBoards. The Master board generates clock and control signals and transmits 8-bit data, while the Slave board receives the data and displays it through LEDs. The Verilog HDL implementation ensures complete control over timing, synchronization, and bit-level communication.

## 2 Objectives

- To design SPI Master and Slave modules using Verilog HDL.
- To establish synchronous communication between two ZedBoards.
- To transfer an 8-bit pattern from Master to Slave.
- To visualize received data using LEDs on the Slave board.
- To verify proper synchronization through simulation and hardware testing.

## 3 Tools Used

- **Software:** Xilinx Vivado Design Suite
- **Language:** Verilog HDL
- **Hardware:** Two Digilent ZedBoards (Zynq-7000 SoC)
- **Additional Tools:** Jumper wires, breadboard, connecting cables

## 4 Description of SPI Protocol

SPI is a four-wire communication protocol comprising:

- **MOSI (Master Out Slave In):** Transmits data from Master to Slave.
- **MISO (Master In Slave Out):** Transmits data from Slave to Master.
- **SCLK (Serial Clock):** Clock signal generated by Master for synchronization.
- **SS (Slave Select):** Active-low signal used to enable a particular Slave.



Figure 1: SPI Communication Between Master and Slave ZedBoards

### 4.1 Working Principle of SPI Communication

The SPI communication in this design works on the concept of clock-driven data exchange, where each bit is shifted in and out synchronously with clock edges.

#### 4.1.1 1. Clock Synchronization

The master ZedBoard generates a serial clock (SCLK) that governs the timing of all data transfers. The slave board uses this clock to sample data on the MOSI line. For this design, SPI Mode 0 (CPOL=0, CPHA=0) is used:

- Clock is idle LOW.
- Data is sampled at the rising edge of SCLK.

#### 4.1.2 2. Data Transmission

When communication begins, the master sets the **Slave Select (SS)** line low to enable the slave. The 8-bit data from the master is loaded into a shift register. With each clock pulse:

- The master shifts out one bit on MOSI.

- The slave reads that bit on the rising edge of SCLK.

After 8 clock cycles, all 8 bits have been transferred. The SS line is then pulled high, marking the end of the frame.

#### 4.1.3 3. Data Reception at the Slave

The slave has an 8-bit shift register synchronized with the master's clock. It continuously shifts in bits from the MOSI line and latches them once the full byte is received. The final 8-bit data is then mapped to LEDs, visually confirming successful data transfer.

#### 4.1.4 4. Master-Slave Synchronization Logic

Both boards contain Finite State Machines (FSMs) that control their operation. **Master FSM:**

- IDLE → LOAD → TRANSFER → COMPLETE

**Slave FSM:**

- IDLE → RECEIVE → DISPLAY

This FSM-based design ensures that communication remains deterministic and error-free.

#### 4.1.5 5. Timing Considerations

SPI timing requires that data be stable before the sampling edge. Therefore, MOSI is updated on the falling edge and read on the rising edge of the clock. This guarantees no setup/hold violations.

## 4.2 Connection Diagram

| Signal | Master (JA Header) | Slave (JA Header) | Direction      |
|--------|--------------------|-------------------|----------------|
| MOSI   | JA1                | JA1               | Master → Slave |
| MISO   | JA2                | JA2               | Slave → Master |
| SCLK   | JA3                | JA3               | Master → Slave |
| SS     | JA4                | JA4               | Master → Slave |
| GND    | GND                | GND               | Common Ground  |

Table 1: Pin Connections Between Master and Slave ZedBoards

# 5 Verilog HDL Implementation

## 5.1 Master Module Code

Listing 1: SPI Master Module

```
1  'timescale 1ns / 1ps
2
3  module spi_master(
4      input  wire  clk,           // System clock (100 MHz)
5      input  wire  reset_n,      // Active-low reset
6      input  wire  start,        // Start signal to initiate SPI
7          transfer
8      input  wire  [7:0] mosi_data, // Data to transmit
9      output reg   [7:0] miso_data, // Data received from slave
10     output reg   done,         // Indicates transfer complete
11     output reg   SCLK,         // SPI Clock
12     output reg   MOSI,         // Master Out Slave In
13     input  wire  MISO,         // Master In Slave Out
14     output reg   SS             // Slave Select (Active Low)
15 );
16
17     reg [3:0] bit_cnt;
18     reg [7:0] shift_reg;
19     reg [2:0] state;
20
21     parameter IDLE = 3'b000,
22             LOAD = 3'b001,
23             TRANSFER = 3'b010,
24             DONE = 3'b011;
25
26
27     always @ (posedge clk or negedge reset_n) begin
28         if (!reset_n) begin
29             state      <= IDLE;
30             SCLK       <= 0;
31             SS         <= 1;
32             MOSI       <= 0;
33             done        <= 0;
34             bit_cnt    <= 0;
35             miso_data  <= 8'd0;
36             shift_reg  <= 8'd0;
37         end else begin
38             case(state)
```

```

34      IDLE: begin
35          done <= 0;
36          SCLK <= 0;
37          SS    <= 1;
38          if(start)
39              state <= LOAD;
40      end
41
42      LOAD: begin
43          SS          <= 0;                      // Activate slave
44          shift_reg <= mosi_data;           // Load data
45          bit_cnt    <= 7;                  // 8 bits total
46          state      <= TRANSFER;
47      end
48
49      TRANSFER: begin
50
51          SCLK <= ~SCLK;
52
53          if(SCLK == 0) begin
54
55              MOSI <= shift_reg[bit_cnt];
56          end else begin
57
58              shift_reg[bit_cnt] <= MISO;
59              if(bit_cnt == 0)
60                  state <= DONE;
61              else
62                  bit_cnt <= bit_cnt - 1;
63          end
64      end
65
66      DONE: begin
67          SCLK       <= 0;
68          SS        <= 1;                   // Deactivate
69          miso_data <= shift_reg;
70          done      <= 1;
71          state     <= IDLE;

```

```

72           end
73       endcase
74   end
75 end
76 endmodule

```

## 5.2 Master Test bench code

Listing 2: SPI Master Test bench Code

```

1  `timescale 1ns/1ps
2  module tb_spi_gate;
3      reg clk, rst, start;
4      reg miso;
5      wire mosi, sclk, cs;
6      wire [7:0] miso_data;
7
8      spi_master_gate uut_master (
9          .clk(clk),
10         .rst(rst),
11         .start(start),
12         .mosi_data(8'b11001100),
13         .miso_data(miso_data),
14         .miso(miso),
15         .mosi(mosi),
16         .sclk(sclk),
17         .cs(cs)
18     );
19
20     spi_slave_gate uut_slave (
21         .sclk(sclk),
22         .cs(cs),
23         .mosi(mosi),
24         .miso(miso)
25     );
26
27     initial clk = 0;
28     always #5 clk = ~clk; // 100 MHz clock
29
30     initial begin

```

```
31         rst = 1; start = 0;
32
33         #20 rst = 0;
34
35         #50 start = 1;
36
37         #10 start = 0;
38
39         #2000;
40
41         $display("Received data: %b", miso_data);
42
43         $finish;
44
45     end
46
47
48 endmodule
```

### 5.3 Master board top level module code

Listing 3: Master board top module code

```

1  `timescale 1ns / 1ps
2
3  module top_spi_master_zedboard(
4      input wire clk,
5      input wire reset_n,
6      input wire start,           // From BTNU
7      input wire [7:0] SW,        // 8 DIP switches
8      output wire MOSI,
9      input wire MISO,
10     output wire SCLK,
11     output wire SS
12 );
13
14     wire [7:0] tx_data;
15     wire [7:0] rx_data;
16     wire done;
17
18     // Send DIP switch values
19     assign tx_data = SW;
20
21     spi_master spi_inst (
22         .clk(clk),
23         .reset_n(reset_n),
24         .start(start),
25         .mosi_data(tx_data),
26         .miso_data(rx_data),
27         .done(done)
28     );

```

```

26     .done(done),
27     .SCLK(SCLK),
28     .MOSI(MOSI),
29     .MISO(MISO),
30     .SS(SS)
31 );
32
33 endmodule

```

## 5.4 Master board Constraints (XDC) File

Listing 4: Master board Constraints file

```

1
2 ## -----
3 ## Clock (100 MHz) - Bank 13 (3.3V)
4 ## -----
5 set_property PACKAGE_PIN Y9 [get_ports clk]
6 set_property IOSTANDARD LVCMOS33 [get_ports clk]
7 create_clock -period 10.0 -name sys_clk -waveform {0 5} [
8     get_ports clk]
9
10 ## -----
11 ## Reset Button (BTNC) - Bank 34 (1.8V)
12 ## -----
13 set_property PACKAGE_PIN P16 [get_ports reset_n]
14 set_property IOSTANDARD LVCMOS18 [get_ports reset_n]
15
16 ## -----
17 ## Start Button (BTNU) - Bank 34 (1.8V)
18 ## -----
19 set_property PACKAGE_PIN R18 [get_ports start]
20 set_property IOSTANDARD LVCMOS18 [get_ports start]
21
22 ## -----
23 ## DIP Switches SW[7:0] - Bank 35 (1.8V)
24 ## -----
25 set_property PACKAGE_PIN F22 [get_ports {SW[0]}]
26 set_property PACKAGE_PIN G22 [get_ports {SW[1]}]
27 set_property PACKAGE_PIN H22 [get_ports {SW[2]}]

```

```

27 set_property PACKAGE_PIN F21 [get_ports {SW[3]}]
28 set_property PACKAGE_PIN H19 [get_ports {SW[4]}]
29 set_property PACKAGE_PIN H18 [get_ports {SW[5]}]
30 set_property PACKAGE_PIN H17 [get_ports {SW[6]}]
31 set_property PACKAGE_PIN M15 [get_ports {SW[7]}]
32 set_property IOSTANDARD LVCMOS18 [get_ports {SW[*]}]

33

34 ## -----
35 ## SPI Interface (PMOD JA) - Bank 13 (3.3V)
36 ## -----
37 # JA1 = SS, JA2 = MOSI, JA3 = MISO, JA4 = SCLK
38 set_property PACKAGE_PIN Y11 [get_ports SS]
39 set_property PACKAGE_PIN AA11 [get_ports MOSI]
40 set_property PACKAGE_PIN Y10 [get_ports MISO]
41 set_property PACKAGE_PIN AA9 [get_ports SCLK]

42

43 set_property IOSTANDARD LVCMOS33 [get_ports SS]
44 set_property IOSTANDARD LVCMOS33 [get_ports MOSI]
45 set_property IOSTANDARD LVCMOS33 [get_ports MISO]
46 set_property IOSTANDARD LVCMOS33 [get_ports SCLK]

47

48 ## -----
49 ## Bank-wide I/O Voltage Declaration
50 ## -----
51 set_property IOSTANDARD LVCMOS33 [get_ports -of_objects [
      get_iobanks 13]]
52 set_property IOSTANDARD LVCMOS18 [get_ports -of_objects [
      get_iobanks 34]]
53 set_property IOSTANDARD LVCMOS18 [get_ports -of_objects [
      get_iobanks 35]]

```

## 5.5 Slave Module Code

Listing 5: SPI Slave Module

```

1 'timescale 1ns / 1ps
2 module spi_slave(
3     input wire          clk,           // System clock (not SPI
4                           clock)
5     input wire          reset_n,       // Active-low reset

```

```

5   input  wire          SCLK ,           // SPI clock from master
6   input  wire          SS ,            // Slave select (active low)
7   input  wire          MOSI ,           // Master Out Slave In
8   output reg           MISO ,           // Master In Slave Out
9   output reg [7:0]     miso_data ,      // Data received from master
10  input  wire [7:0]    mosi_data ,      // Data to send back
11  output reg           done ,           // Indicates data reception
12   complete
13 );
14
15   reg [7:0] shift_in;
16   reg [7:0] shift_out;
17   reg [2:0] bit_cnt;
18   reg      sclk_prev;
19
20   always @ (posedge clk or negedge reset_n) begin
21     if (!reset_n) begin
22       shift_in <= 8'd0;
23       shift_out <= 8'd0;
24       miso_data <= 8'd0;
25       bit_cnt <= 3'd0;
26       MISO <= 1'b0;
27       done <= 1'b0;
28       sclk_prev <= 1'b0;
29     end else begin
30       sclk_prev <= SCLK;
31       if (!SS) begin
32         done <= 1'b0;
33         if (sclk_prev == 0 && SCLK == 1) begin
34           // Rising edge: sample MOSI
35           shift_in <= {shift_in[6:0], MOSI};
36           bit_cnt <= bit_cnt + 1;
37         end
38         else if (sclk_prev == 1 && SCLK == 0) begin
39           // Falling edge: drive MISO
40           MISO <= shift_out[7];
41           shift_out <= {shift_out[6:0], 1'b0};
42         end
43         if (bit_cnt == 3'd7 && sclk_prev == 0 && SCLK ==
44           1) begin
45           miso_data <= {shift_in[6:0], MOSI};

```

```

42          done <= 1'b1;
43          bit_cnt <= 0;
44          shift_out <= mosi_data; // Prepare next byte
45          to send
46      end
47  end else begin
48      // When slave not selected
49      bit_cnt <= 0;
50      done <= 0;
51      MISO <= 0;
52      shift_out <= mosi_data;
53  end
54 end
55 endmodule

```

## 5.6 Slave test bench Code

Listing 6: SPI Slave Test bench code

```

1 'timescale 1ns/1ps
2 module tb_spi_slave_gate;
3     reg sys_clk;
4     reg reset_n;
5     reg sclk;
6     reg ss;
7     reg mosi;
8     wire miso;
9     wire [7:0] miso_data;
10    reg [7:0] master_tx_data;
11    reg [7:0] slave_tx_data;
12    integer i;
13    spi_slave uut_slave (
14        .clk(sys_clk),
15        .reset_n(reset_n),
16        .SCLK(sclk),
17        .SS(ss),
18        .MOSI(mosi),
19        .MISO(miso),
20        .miso_data(miso_data),

```

```

21     .mosi_data(slave_tx_data),
22     .done()
23 );
24 initial sys_clk = 0;
25 always #5 sys_clk = ~sys_clk; // 100 MHz
26 initial begin
27     sclk = 0;
28     ss   = 1;
29     mosi = 0;
30     master_tx_data = 8'b1010_1100;
31     slave_tx_data  = 8'b0101_1010;
32     reset_n = 0;
33     #50;
34     reset_n = 1;
35     #100;
36     ss = 0;
37     $display("Starting SPI transfer at time %t", $time);
38     for (i = 7; i >= 0; i = i - 1) begin
39         mosi = master_tx_data[i];
40         #50 sclk = 1;
41         #50 sclk = 0;
42         $display("Bit %0d sent = %b, MISO = %b", i, mosi,
43                   miso);
44     end
45     s = 1;
46     #100;
47     $display("Transfer complete at %t", $time);
48     $display("Slave received (miso_data) = %b", miso_data);
49     #100;
50     $finish;
51 end
endmodule

```

## 5.7 Slave board top level module code

Listing 7: SPI Slave top level Module code

```

1 `timescale 1ns / 1ps
2 module top_spi_slave (

```

```

3      input wire          clk,           // 100 MHz system clock from
4          board
5      input wire          reset_n,        // Active-low reset (
6          pushbutton)
7      input wire          SCLK,          // SPI clock from master
8      input wire          SS,            // Slave Select (active low)
9      input wire          MOSI,          // Master Out Slave In
10     output wire         MISO,          // Master In Slave Out
11     output wire [7:0]    LD             // LEDs to show received data
12 );
13
14     wire [7:0] miso_data;
15     wire       done;
16     reg  [7:0] mosi_data_reg = 8'h5A;
17     spi_slave u_spi_slave (
18         .clk(clk),
19         .reset_n(reset_n),
20         .SCLK(SCLK),
21         .SS(SS),
22         .MOSI(MOSI),
23         .MISO(MISO),
24         .miso_data(miso_data),
25         .mosi_data(mosi_data_reg),
26         .done(done)
27 );
28
29     assign LD = miso_data;
30
31 endmodule

```

## 5.8 Slave board Constraints(XDC) File

Listing 8: SPI Slave Board Constraints File

```

1 # =====
2 # System Clock - 3.3V, Bank 13
3 # =====
4 set_property PACKAGE_PIN Y9 [get_ports clk]
5 set_property IOSTANDARD LVCMOS18 [get_ports clk]
6 create_clock -period 10.0 -name sys_clk -waveform {0 5} [
7     get_ports clk]
8 # =====

```

```

9 # Reset Button - 1.8V, Bank 34
10 # =====
11
12 set_property PACKAGE_PIN P16 [get_ports reset_n]
13 set_property IOSTANDARD LVCMOS18 [get_ports reset_n]
14
15 # =====
16 # SPI Interface on PMOD JA - Bank 35, 1.8V
17 # =====
18
19 set_property PACKAGE_PIN AA9 [get_ports SCLK]
20 set_property IOSTANDARD LVCMOS18 [get_ports SCLK]
21
22 set_property PACKAGE_PIN AA11 [get_ports MOSI]
23 set_property IOSTANDARD LVCMOS18 [get_ports MOSI]
24
25 set_property PACKAGE_PIN Y10 [get_ports MISO]
26 set_property IOSTANDARD LVCMOS18 [get_ports MISO]
27
28 set_property PACKAGE_PIN Y11 [get_ports SS]
29 set_property IOSTANDARD LVCMOS18 [get_ports SS]
30
31 # =====
32 # User LEDs - Bank 33
33 # =====
34
35 set_property PACKAGE_PIN T22 [get_ports {LD[0]}]; # "LD0"
36 set_property IOSTANDARD LVCMOS33 [get_ports LD[0]]
37
38 set_property PACKAGE_PIN T21 [get_ports {LD[1]}]; # "LD1"
39 set_property IOSTANDARD LVCMOS33 [get_ports LD[1]]
40
41 set_property PACKAGE_PIN U22 [get_ports {LD[2]}]; # "LD2"
42 set_property IOSTANDARD LVCMOS33 [get_ports LD[2]]
43
44 set_property PACKAGE_PIN U21 [get_ports {LD[3]}]; # "LD3"
45 set_property IOSTANDARD LVCMOS33 [get_ports LD[3]]
46
47 set_property PACKAGE_PIN V22 [get_ports {LD[4]}]; # "LD4"

```

```

48 set_property IOSTANDARD LVCMOS33 [get_ports LD[4]]
49
50 set_property PACKAGE_PIN W22 [get_ports {LD[5]}]; # "LD5"
51 set_property IOSTANDARD LVCMOS33 [get_ports LD[5]]
52
53 set_property PACKAGE_PIN U19 [get_ports {LD[6]}]; # "LD6"
54 set_property IOSTANDARD LVCMOS33 [get_ports LD[6]]
55
56 set_property PACKAGE_PIN U14 [get_ports {LD[7]}]; # "LD7"
57 set_property IOSTANDARD LVCMOS33 [get_ports LD[7]]

```

## 5.9 Schematic diagram of Master and Slave Communications



Figure 2: Schematic of SPI Master Communication



Figure 3: Schematic of SPI Slave Communication

## 6 ZedBoard Implementation



Figure 4: SPI Master Implementation on ZedBoard



Figure 5: SPI Slave Implementation on ZedBoard

## 7 Results and Discussion

- The SPI protocol was successfully established between two ZedBoards.
- The Master transmitted 8-bit data patterns accurately.
- The Slave displayed corresponding LED patterns with no data loss.
- Timing waveforms verified synchronization between MOSI, SCLK, and SS.



Figure 6: Simulation Waveform of SPI Master Communication



Figure 7: Simulation Waveform of SPI Slave Communication

## 8 Advantages

- High-speed synchronous data transfer.
- Simple and efficient design using minimal FPGA resources.
- Supports full-duplex communication.
- Easily scalable for multi-slave systems.

## 9 Conclusion

The SPI protocol was successfully designed and implemented using two ZedBoards in a Master-Slave configuration. The system efficiently transferred an 8-bit pattern and displayed it on the Slave board's LEDs, confirming reliable data exchange. The implementation verified the fundamental working of SPI communication and demonstrated FPGA-based protocol realization.

## References

1. Digilent ZedBoard Reference Manual.
2. Xilinx Vivado Design Suite User Guide.
3. Motorola SPI Bus Specification.
4. P. P. Chu, "FPGA Prototyping by Verilog Examples," Wiley Publications.