

# SPI Master–Slave RTL Design (4-Mode Configurable)

## Objective

Design Verilog RTL modules for a **configurable SPI Master and SPI Slave**, supporting all **four SPI modes** determined by **CPOL** (clock polarity) and **CPHA** (clock phase).

Each module must follow the step-by-step flow-chart behavior given below and implement it as a clean **Finite State Machine (FSM)**.

---

## SPI MASTER

### Functional Overview

Implements full-duplex serial communication with one or more slaves over:

- **MOSI** (Master Out Slave In)
- **MISO** (Master In Slave Out)
- **SCLK** (Serial Clock)
- **CS\_N** (Chip Select, active-low)

Supports:

- 4 SPI modes:

| Mode | CPOL | CPHA | Description                                  |
|------|------|------|----------------------------------------------|
| 0    | 0    | 0    | Sample on rising edge, shift on falling edge |
| 1    | 0    | 1    | Shift on rising, sample on falling           |
| 2    | 1    | 0    | Sample on falling, shift on rising           |
| 3    | 1    | 1    | Shift on falling, sample on rising           |

- Configurable bit length (default: 8 bits)

- MSB-first or LSB-first shifting
- 

## FSM Design (based on flow chart)

| State       | Description / Actions                                                                                                            |
|-------------|----------------------------------------------------------------------------------------------------------------------------------|
| IDLE        | CS_N = 1 (inactive); <code>busy=0</code> . Wait for <code>start</code> = 1.                                                      |
| ASSERT_CS   | Pull CS_N = 0, optionally delay 1 cycle. Initialize bit counter.                                                                 |
| LOAD        | Load <code>tx_reg</code> with transmit data; reset bit counter = <code>N-1</code> .                                              |
| SHIFT       | Generate SCLK edges as per CPHA/CPOL. On each active edge:- Drive MOSI (MSB/LSB)- Sample MISO- Shift register- Decrement counter |
| CHECK_DONE  | If bit_counter $\geq 0 \rightarrow$ go back to SHIFT; else $\rightarrow$ next.                                                   |
| DEASSERT_CS | Pull CS_N = 1 after optional delay.                                                                                              |
| DONE        | Latch received data into <code>rx_reg</code> , clear busy, pulse <code>done=1</code> . Return to IDLE.                           |

---

## Expected Inputs

| Signal                      | Description                               |
|-----------------------------|-------------------------------------------|
| <code>clk_sys</code>        | System clock                              |
| <code>rst</code>            | Reset                                     |
| <code>start</code>          | Start transfer pulse                      |
| <code>data_in[N-1:0]</code> | Data to transmit                          |
| <code>cpol</code>           | Clock polarity select                     |
| <code>cpha</code>           | Clock phase select                        |
| <code>msb_first</code>      | Bit-order select                          |
| <code>bit_len</code>        | Configurable number of bits (default = 8) |
| <code>miso</code>           | Serial data from slave                    |

## Expected Outputs

| Signal            | Description     |
|-------------------|-----------------|
| <code>mosi</code> | Master data out |

| Signal          | Description                         |
|-----------------|-------------------------------------|
| sclk            | Serial clock generated by master    |
| cs_n            | Active-low chip select              |
| data_out[N-1:0] | Received data                       |
| busy            | High during transmission            |
| done            | Single-cycle pulse after completion |

---

## Timing Notes

- For CPHA = 0, the **first data bit** must be ready before the first SCLK edge.
  - For CPHA = 1, the first data bit is launched on the first edge.
  - SCLK toggles according to CPOL:
    - CPOL = 0 → idle low, active high.
    - CPOL = 1 → idle high, active low.
- 

## SPI SLAVE

### Functional Overview

Responds to a master transaction when its **CS\_N is low**.

- Shifts data on SCLK edges based on CPOL/CPHA.
  - Receives MOSI data and transmits MISO data simultaneously.
  - Uses same bit length and bit order as master.
- 

### FSM Design

| State      | Description / Actions                                                                           |
|------------|-------------------------------------------------------------------------------------------------|
| IDLE       | Wait for CS_N = 1 → 0 (slave selected).                                                         |
| LOAD       | Load tx_reg with outgoing data; clear bit_cnt .                                                 |
| SHIFT_RXTX | For each SCLK edge:- Sample MOSI at correct edge.- Shift out next MISO bit.- Decrement counter. |

| State                | Description / Actions                                                       |
|----------------------|-----------------------------------------------------------------------------|
| <b>CHECK_DONE</b>    | If counter $\geq 0 \rightarrow$ continue shifting; else $\rightarrow$ next. |
| <b>LATCH_DATA</b>    | Store received data, raise <code>rx_valid=1</code> .                        |
| <b>WAIT_DEASSERT</b> | Wait for CS_N = 1 (deselected). Go to IDLE.                                 |

## Expected Inputs

| Signal            | Description                     |
|-------------------|---------------------------------|
| <code>sclk</code> | Clock from master               |
| <code>cs_n</code> | Chip select from master         |
| <code>mosi</code> | Serial data from master         |
| <code>rst</code>  | Reset                           |
| <code>cpol</code> | Clock polarity (matches master) |
| <code>cpha</code> | Clock phase (matches master)    |

## Expected Outputs

| Signal                      | Description                       |
|-----------------------------|-----------------------------------|
| <code>miso</code>           | Data back to master               |
| <code>rx_data[N-1:0]</code> | Received byte                     |
| <code>rx_valid</code>       | Pulse high when new byte received |

## Integration Requirements

- Connect **Master  $\leftrightarrow$  Slave** signals:
  - `MOSI`  $\rightarrow$  `MOSI`, `MISO`  $\leftarrow$  `MISO`, `SCLK`  $\rightarrow$  `SCLK`, `CS_N`  $\rightarrow$  `CS_N`.
- Verify all 4 SPI modes by sweeping CPOL/CPHA.
- Simulate single and multiple transfers.
- Waveforms must show correct bit alignment and edge relationships.

## Deliverables

1. `spi_master.v` — Master RTL (FSM per flow chart)
2. `spi_slave.v` — Slave RTL (FSM per flow chart)
3. `spi_master_slave_tb.v` — Testbench with connected master–slave pair
  - Test all four (CPOL, CPHA) combinations
  - Show MOSI/MISO/SCLK timing
  - Validate full-duplex exchange