

# Project Report: 16-bit SPI Interface State Machine

A Digital Logic Designer

December 4, 2025

## Abstract

This report details the design, implementation, and verification of a 16-bit Serial Peripheral Interface (SPI) master transmitter implemented in Verilog HDL. The design utilizes a three-state Finite State Machine (FSM) to manage the serialization of 16-bit input data and drive the standard SPI signals: Chip Select (CS), Serial Clock (SCLK), and Master Out Slave In (MOSI). The implemented FSM states are IDLE, TRANSFER, and DONE. Verification was performed using a dedicated testbench to transmit a sequence of hexadecimal data words.

## 1 Introduction

The Serial Peripheral Interface (SPI) is a synchronous serial communication interface specification commonly used for short-distance, master-slave communication in embedded systems. This project focuses on developing the digital logic for an SPI master transmitter capable of sending 16 bits of data per transaction. The core of the implementation is a state machine that controls the timing and sequencing of the data transmission.

## 2 Design Specification

### 2.1 Interface Ports

The top-level module, SPI\_state, exposes the following ports:

- clk: System clock input.
- rst: Asynchronous active-low reset.
- data\_in [15:0]: The 16-bit data word to be transmitted.
- spi\_cs: Output, Chip Select (active low).
- spi\_sclk: Output, Serial Clock.
- spi\_data: Output, Master Out Slave In (MOSI).
- counter [4:0]: Output, 5-bit counter for tracking bit position.

### 2.2 State Machine Logic

The FSM is defined by three local parameters: IDLE = 0, TRANSFER = 1, and DONE = 2. The transitions are governed by the system clock (clk) and the internal counter (count). The state transitions are as follows:

1. **IDLE State (00):**

- Initializes sclk to logic low ('0').
- cs is asserted high (1) if count is 16 (indicating the start of a new, complete transaction), otherwise it is kept low.
- The FSM transitions to the TRANSFER state.

## 2. TRANSFER State (01):

- sclk is kept low ('0').
- cs is asserted low ('0') to keep the slave selected.
- The MOSI line is driven with the current data bit: data\_in [count - 1].
- The count register is decremented (count  $\leftarrow$  count - 1).
- The FSM transitions to the DONE state.

## 3. DONE State (10):

- sclk is asserted high ('1'), completing the clock cycle and latching the bit on the slave side (assuming CPOL=0, CPHA=1).
- If count > 0, there are more bits to send, and the FSM transitions back to the IDLE state to prepare the next bit.
- If count = 0, all 16 bits have been sent. The count is reset to 16, and the FSM transitions back to IDLE to wait for the next data word.

## 3 Implementation (Verilog HDL)

The design is implemented in the Verilog module SPI\_state.v. The full code is provided in Appendix A.1.

### 3.1 Code Snippet: State Logic

The heart of the design is the clocked sequential logic, as shown in the case statement below:

```

1  else begin
2      case (state)
3          IDLE: begin
4              sclk  <= 1'b0;
5              cs    <= count == 16;
6              state <= TRANSFER;
7          end
8
9          TRANSFER: begin
10             sclk  <= 1'b0;
11             cs    <= 1'b0;
12             MOSI  <= data_in [count - 1];
13             count <= count - 1;
14             state <= DONE;
15         end
16
17         DONE: begin
18             sclk <= 1'b1;
19             if (count > 0) state <= IDLE;
20             else begin
21                 count <= 16;
22                 state <= IDLE;
23             end
24         end

```

```

25           default: state <= IDLE;
26       endcase
27   end

```

Listing 1: Core State Machine Logic from SPI\_state.v

## 4 Verification and Testbench

### 4.1 Testbench Description

Verification was performed using the spi\_state\_tb.v testbench. The testbench instantiated the Device Under Test (dut) and generated the required stimulus signals.

- **Clock Generation:** A clk signal with a period of 10ns (5ns high, 5ns low) was generated.
- **Reset Sequence:** The rst signal was held low initially and asserted high after 10ns.
- **Data Input:** A sequence of 16-bit hexadecimal values was applied to the data\_in port, with each new word applied after a period of 480ns, which is sufficient time to complete a full 16-bit transmission (16bits × 3clocks/bit × 10ns/clock ≈ 480ns).

The sequence of transmitted data words is: 16'hA569, 16'h2563, 16'h9B63, 16'h6A61, 16'hA265, and 16'h7564.

### 4.2 Testbench Code Snippet

```

1 initial begin
2     #10 rst=1;
3
4     data_in = 16'hA569;
5     #480 data_in = 16'h2563;
6     #480 data_in = 16'h9B63;
7     #480 data_in = 16'h6A61;
8     #480 data_in = 16'hA265;
9     #480 data_in = 16'h7564;
10
11    $finish;
12 end

```

Listing 2: Data Stimulus from spi\_state\_tb.v

## 5 Conclusion

The 16-bit SPI master transmitter was successfully designed using a three-state FSM (IDLE, TRANSFER, DONE). The design ensures that each of the 16 bits of data is correctly outputted on the spi\_data (MOSI) line, synchronized with the spi\_sclk signal, and contained within the assertion period of the spi\_cs signal. The testbench demonstrated the transmission of multiple data packets, confirming the functional correctness of the design.

## A Verilog Source Code

### A.1 SPI\_state.v

```
1 module SPI_state (
2     input    wire      clk,
3     input    wire      rst,
4     input    wire [15:0] data_in,
5
6     output   wire      spi_cs,
7     output   wire      spi_sclk,
8     output   wire      spi_data, // MISO
9     output   wire [04:0] counter
10 );
11
12 localparam [1:0] IDLE      = 0;
13 localparam [1:0] TRANSFER  = 1;
14 localparam [1:0] DONE       = 2;
15
16 //reg [15:0] MOSI;
17 reg        MOSI;
18 reg [04:0] count;
19 reg        cs;
20 reg        sclk;
21 reg [01:0] state;
22
23 always @ (posedge clk or negedge rst) begin
24     if (~rst) begin
25         MOSI  <= 1'b0;
26         count <= 05'd16;
27         cs    <= 01'b1;
28         sclk <= 01'b0;
29         state <= IDLE;
30     end
31
32     else begin
33         case (state)
34             IDLE: begin
35                 sclk <= 1'b0;
36                 cs    <= count == 16;
37                 state <= TRANSFER;
38             end
39
40             TRANSFER: begin
41                 sclk <= 1'b0;
42                 cs    <= 1'b0;
43                 MOSI  <= data_in [count - 1];
44                 count <= count - 1;
45                 state <= DONE;
46             end
47
48             DONE: begin
49                 sclk <= 1'b1;
50                 if (count > 0) state <= IDLE;
51                 else begin
52                     count <= 16;
53                     state <= IDLE;
54                 end
55             end
56
57             default: state <= IDLE;
58         endcase
59     end
60 
```

```

60 end
61
62 assign spi_cs    = cs;
63 assign spi_sclk = sclk;
64 assign spi_data = MOSI;
65 assign counter = count;
66
67 endmodule

```

Listing 3: SPI State Machine Module (SPI\_state.v)

## A.2 spi\_state\_tb.v

```

1 `timescale 1ns / 1ps
2
3 module spi_state_tb;
4
5 reg      clk;
6 reg      rst;
7 reg [15:0] data_in;
8
9 wire     spi_cs;
10 wire    spi_sclk;
11 wire    spi_data;
12 wire [04:0] counter;
13
14 SPI_state dut (
15   .clk (clk),
16   .rst (rst),
17   .counter (counter),
18   .data_in (data_in),
19   .spi_cs (spi_cs),
20   .spi_sclk (spi_sclk),
21   .spi_data (spi_data)
22 );
23
24 initial begin
25   clk      = 0;
26   rst      = 0;
27   data_in = 0;
28 end
29
30 always #5 clk = ~clk;
31
32 initial begin
33   #10 rst=1;
34
35   data_in = 16'hA569;
36   #480 data_in = 16'h2563;
37   #480 data_in = 16'h9B63;
38   #480 data_in = 16'h6A61;
39   #480 data_in = 16'hA265;
40   #480 data_in = 16'h7564;
41
42   $finish;
43 end
44
45 endmodule

```

Listing 4: SPI State Machine Testbench (spi\_state\_tb.v)