Building a Custom SPI Slave Module in Verilog
I recently implemented a Serial Peripheral Interface (SPI) slave module targeting the Arty S7-25 FPGA. This project taught me a lot about both SPI protocol implementation and FPGA development. Here’s a deep dive into the design and key learnings.
Core Features
- Full SPI slave interface implementation
- 24-bit transaction support
- Multiple addressable registers
- Built-in debugging capabilities
- LED status indicators
- 10-bit debug bus
Design Architecture
Protocol Implementation
The module handles 24-bit transactions that include:
- 10-bit address space
- 3-bit command opcode (010 for Write, 011 for Read)
- 8-bit data payload
- Additional padding bits
Register Map
The module implements several addressable registers:
1
2
3
4
reg_015 (0x15) - General purpose register
reg_215 (0x215) - Extended address register
reg_012 (0x12) - General purpose register
reg_A (0xA) - General purpose register
Key Components
SPI Interface
1
2
3
4
input i_SPI_CS, // Chip Select
input i_SPI_CLK, // SPI Clock
input MOSI_in, // Master Out Slave In
output MISO_out // Master In Slave Out
Shift Register Logic
The heart of the module is a 24-bit shift register that handles incoming data on the MOSI line. It’s synchronized to the SPI clock and handles data capture on positive edges:
1
2
3
4
5
6
7
8
9
10
11
12
13
always @(posedge i_SPI_CLK, posedge i_SPI_CS) begin
if (i_SPI_CS)
begin
bits <= 0;
counter <= 0;
end
else
begin
bits <= bits << 1;
bits[0] <= MOSI_in;
counter <= counter + 1;
end
end
Debug Features
LED Indicators
- Green LED: Switch enabled
- Red LED: Switch disabled
Debug Bus
A 10-bit debug bus provides real-time visibility into:
- Free-running counter status
- MISO output state
- Chip select status
- Address matching
- Data validation
- Command decoding
Interesting Challenges
Clock Domain Crossing: Handling the transition between SPI clock and FPGA system clock domains required careful consideration of synchronization.
Command Timing: Ensuring proper timing for read/write operations while maintaining protocol compliance was tricky. The solution involved careful state management around the chip select signal.
Register Access: Implementing clean register read/write logic while maintaining proper synchronization with the SPI clock required several iterations.
Future Improvements
- Add FIFO buffers for improved data handling
- Implement error detection
- Add configuration registers for flexible operation
- Enhance debug capabilities with transaction counters
- Add support for different SPI modes
Lessons Learned
- Always add more debug signals than you think you’ll need
- Careful state machine design is crucial for reliable operation
- Simulation saves hours of hardware debugging time
- Clock domain crossing requires special attention
- LED indicators are invaluable for basic debugging
Conclusion
This project provided valuable hands-on experience with both SPI protocol implementation and FPGA development. The modular design approach made debugging easier and will allow for future enhancements. Looking forward to expanding its capabilities in future iterations!
Stay tuned for more FPGA development adventures!