《深入解析SPI协议及其FPGA高效实现》-- 第二篇:SPI控制器FPGA架构设计

article/2025/7/12 19:25:55

第二篇:SPI控制器FPGA架构设计

聚焦模块化设计、时序优化与资源管理

1. 系统级架构设计

1.1 模块化硬件架构

verilog

module spi_controller (input  wire        clk,          // 系统时钟 (100 MHz)input  wire        rst_n,        // 异步复位// 配置接口input  wire [15:0] clk_div,      // 时钟分频系数input  wire [1:0]  spi_mode,     // CPOL/CPHA模式input  wire        lsb_first,    // 位传输顺序// 数据接口input  wire [7:0]  tx_data,      // 发送数据output reg  [7:0]  rx_data,      // 接收数据output reg         busy,         // 传输状态// SPI物理接口output wire        sck,          // SPI时钟output wire        mosi,         // 主出从入input  wire        miso,         // 主入从出output wire        cs_n          // 片选 (低有效)
);
// 核心子模块实例化
clock_gen     u_clk_gen(.*);      // 时钟生成
fsm_controller u_fsm(.*);         // 状态机
data_path     u_datapath(.*);     // 数据路径
cs_decoder    u_cs_dec(.*);       // 片选译码
endmodule

关键模块说明

  • 时钟域划分
    高速域 (clk) :配置寄存器访问 (100MHz)
    SPI域 (spi_clk) :SCK同步逻辑 (≤50MHz)
    跨时钟域同步 :TX/RX数据通过异步FIFO交换
1.2 寄存器映射表
地址偏移寄存器名位定义功能
0x00CTRL[0]:使能 [1]:LSB [3:2]:模式控制寄存器
0x04CLK_DIV16位分频系数时钟分频
0x08TX_DATA8位发送数据发送缓冲区
0x0CRX_DATA8位接收数据接收缓冲区
0x10STATUS[0]:忙 [1]:传输完成状态寄存器

2. 时钟生成电路

2.1 可编程分频器设计

verilog

module clock_gen (input  wire clk,input  wire [15:0] div_value,output reg  spi_clk
);reg [15:0] counter = 0;reg clk_phase = 0;  // 相位控制always @(posedge clk) beginif (counter >= div_value) begincounter <= 0;clk_phase <= ~clk_phase;  // 翻转相位end else begincounter <= counter + 1;endend// CPOL控制输出极性assign sck = (spi_mode[1]) ? ~clk_phase : clk_phase;
endmodule

关键技术

  • 小数分频实现 :通过累加器实现N.5分频(如3.5分频)
    verilog
  reg [3:0] acc = 0;always @(posedge clk) beginacc <= acc + 4'd2;         // 步进值=2 (2/4=0.5)spi_clk <= (acc < 4'd4);   // 50%占空比end
  • 动态时钟切换 :CPOL变化时插入死区时间避免毛刺
    verilog
  always @(posedge clk) beginif (mode_changed) beginsck <= 1'bz;            // 高阻态保持10ns#10; sck <= new_polarity;    // 应用新极性endend

3. 核心状态机设计

3.1 状态转移图

在这里插入图片描述

3.2 Verilog实现

verilog

typedef enum {IDLE, START, TX_RX, STOP} spi_state;
spi_state current_state = IDLE;
reg [2:0] bit_cnt = 0;  // 位计数器always @(posedge spi_clk) begincase(current_state)IDLE: if (cs_active) begincurrent_state <= START;sck <= 1'b0;  // Mode 0初始化endSTART: current_state <= TX_RX;TX_RX:if (bit_cnt == 7) current_state <= STOP;elsebit_cnt <= bit_cnt + 1;STOP:if (!cs_hold) current_state <= IDLE;endcase
end

关键特性

  • CPHA自适应 :通过状态机控制采样沿
    verilog
  // CPHA=0: 上升沿采样,下降沿切换数据wire sample_edge = (spi_mode[0]) ? negedge sck : posedge sck;wire shift_edge  = (spi_mode[0]) ? posedge sck : negedge sck;
  • 连续传输支持cs_hold信号保持状态机在STOP状态

4. 数据路径优化

4.1 双缓冲机制

verilog

reg [7:0] tx_buffer, tx_shift;
reg [7:0] rx_shift, rx_buffer;// 发送双缓冲
always @(posedge clk) beginif (!busy) tx_shift <= tx_buffer;  // 空闲时加载新数据
end// 接收双缓冲
always @(posedge spi_clk) beginif (bit_cnt == 0 && current_state == TX_RX)rx_buffer <= rx_shift;  // 传输完成锁存数据
end
4.2 循环移位寄存器

verilog

// LSB优先传输实现
always @(shift_edge) beginif (current_state == TX_RX) begintx_shift <= lsb_first ? {1'b0, tx_shift[7:1]} : {tx_shift[6:0], 1'b0};rx_shift <= lsb_first ? {miso, rx_shift[7:1]} : {rx_shift[6:0], miso};end
end
4.3 多从机数据隔离

verilog

// 基于CS的三态控制
assign mosi = (cs_active) ? tx_shift[7] : 1'bz;
assign miso = (cs_active) ? slave_miso : 1'bz;// 从设备选择译码
module cs_decoder (input  wire [3:0] slave_sel,   // 4位从机选择output reg  [7:0] cs_n         // 8个CS信号
);always @(*) begincs_n = 8'hFF;              // 默认全关if (slave_sel < 8) cs_n[slave_sel] = 1'b0;  // 激活选中从机end
endmodule

5. 时序收敛关键策略

5.1 多周期路径约束

tcl

# XDC约束示例 (Vivado)
set_multicycle_path 2 -setup -from [get_clocks clk] -to [get_clocks spi_clk]
set_multicycle_path 1 -hold -end
5.2 跨时钟域同步链

verilog

// 异步信号三级同步
reg [2:0] sync_miso;
always @(posedge spi_clk) beginsync_miso <= {sync_miso[1:0], miso};
end
5.3 关键路径流水化

verilog

// 添加流水线提升时序
always @(posedge spi_clk) begin// 第1拍:计算下一状态next_state <= fsm_logic(current_state); // 第2拍:更新状态current_state <= next_state;  
end

6. 资源优化技术

6.1 动态部分重配置(Xilinx FPGA)

tcl

# 重配置命令(切换SPI模式)
write_cfgmem -format BIN -interface SPIx4 -size 8 -loadbit "up 0x0 new_mode.bin"
6.2 引脚复用技术
复用方案实现方式节省IO
QSPI模式将WP/HOLD引脚用作数据线2根
三线SPIMOSI/MISO共享SIO线1根
分时复用用同一组引脚驱动多组SPI外设50%

附录:Artix-7资源占用报告

模块LUTsFFs最大频率功耗
时钟生成器4232450 MHz8 mW
状态机7864350 MHz12 mW
数据路径10580400 MHz15 mW
总计225176350 MHz35 mW

http://www.hkcw.cn/article/rRYkeUfyoC.shtml

相关文章

rabbitmq Fanout交换机简介

给每个服务创建一个队列&#xff0c;然后每个业务订阅一个队列&#xff0c;进行消费。 如订单服务起个多个服务&#xff0c;代码是一样的&#xff0c;消费的也是同一个队列。加快了队列中的消息的消费速度。 可以看到两个消费者已经在消费了

Ⅱ.计算机二级选择题(运算符与表达式)

【注&#xff1a;重点题以及添加目录格式导航&#xff01;&#xff01;&#xff01;】 【重点题】&#xff08;第5题&#xff09; 【重点题】&#xff08;第18题&#xff09; 【重点题】&#xff08;第19题&#xff09; 【重点题】&#xff08;第35题&#xff09; 【重点题】&a…

使用Mathematica观察多形式根的分布随参数的变化

有两种方式观察多项式的根随着参数变化&#xff1a;&#xff08;1&#xff09;直接制作一个小的动态视频&#xff1b;&#xff08;2&#xff09;绘制所有根形成的痕迹&#xff08;locus&#xff09;。 制作动态视频&#xff1a; (*Arg-plane plotting routine with plotting …

腾答知识竞赛系统功能介绍

支持抢答题的局域网现场大屏知识竞赛抢答软件&#xff0c;无需网络只要有局域网或者WIFI就可以使用,现场大屏幕显示题目&#xff0c;支持基础题、抢答题、必答题、风险题等题目。 系统支持任何个人或者企业单位使用&#xff0c;使用无人员限制&#xff0c;可放心使用。 抢答时…

Python-matplotlib库之核心对象

matplotlib库之核心对象 FigureFigure作用Figure常用属性Figure常用方法Figure对象的创建隐式创建&#xff08;通过 pyplot&#xff09;显式创建使用subplots()一次性创建 Figure 和 Axes Axes&#xff08;绘图区&#xff09;Axes创建方式Axes基本绘图功能Axes绘图的常用参数Ax…

04powerbi-度量值-筛选引擎CALCULATE()

1、calculate calculate 的参数分两部分&#xff0c;分别是计算器和筛选器 2、多条件calculater与表筛选 多条件有不列的多条件 相同列的多条件 3、calculatertable &#xff08;表&#xff0c;筛选条件&#xff09;表筛选 与calculate用法一样&#xff0c;可以用创建表&…

深度学习原理与Pytorch实战

深度学习原理与Pytorch实战 第2版 强化学习人工智能神经网络书籍 python动手学深度学习框架书 TransformerBERT图神经网络&#xff1a; 技术讲解 编辑推荐 1.基于PyTorch新版本&#xff0c;涵盖深度学习基础知识和前沿技术&#xff0c;由浅入深&#xff0c;通俗易懂&#xf…

LabelImg: 开源图像标注工具指南

LabelImg: 开源图像标注工具指南 1. 简介 LabelImg 是一个图形化的图像标注工具&#xff0c;使用 Python 和 Qt 开发。它是目标检测任务中最常用的标注工具之一&#xff0c;支持 PASCAL VOC 和 YOLO 格式的标注输出。该工具开源、免费&#xff0c;并且跨平台支持 Windows、Lin…

React---day6、7

6、组件之间进行数据传递 **6.1 父传子&#xff1a;**props传递属性 父组件&#xff1a; <div><ChildCpn name"蒋乙菥" age"18" height"1,88" /> </div>子组件&#xff1a; export class ChildCpn extends React.Component…

LLM模型量化从入门到精通:Shrink, Speed, Repeat

前言 神经网络把它们的知识都存成数字啦&#xff0c;主要是训练时学到的权重&#xff0c;还有运行时在每一层流动的激活值。这些数字必须保持在一个固定的数值格式里&#xff0c;而选的格式就决定了每个参数要占多少内存。要是用默认的32位浮点表示&#xff0c;一个有70亿参数…

PHP舆情监控分析系统(9个平台)

PHP舆情监控分析系统&#xff08;9个平台&#xff09; 项目简介 基于多平台热点API接口的PHP实时舆情监控分析系统&#xff0c;无需数据库&#xff0c;直接调用API实时获取各大平台热点新闻&#xff0c;支持数据采集、搜索和可视化展示。 功能特性 &#x1f504; 实时监控 …

贪心算法应用:多重背包启发式问题详解

贪心算法应用&#xff1a;多重背包启发式问题详解 多重背包问题是经典的组合优化问题&#xff0c;也是贪心算法的重要应用场景。本文将全面深入地探讨Java中如何利用贪心算法解决多重背包问题。 多重背包问题定义 **多重背包问题(Multiple Knapsack Problem)**是背包问题的变…

AI预测3D新模型百十个定位预测+胆码预测+去和尾2025年6月2日第96弹

从今天开始&#xff0c;咱们还是暂时基于旧的模型进行预测&#xff0c;好了&#xff0c;废话不多说&#xff0c;按照老办法&#xff0c;重点8-9码定位&#xff0c;配合三胆下1或下2&#xff0c;杀1-2个和尾&#xff0c;再杀4-5个和值&#xff0c;可以做到100-300注左右。 (1)定…

布隆过滤器

文章目录 布隆过滤器&#xff08;Bloom Filter&#xff09;详解&#xff1a;原理、实现与应用场景一、引言二、布隆过滤器的基本原理1. 数据结构2. 插入操作3. 查询操作4. 误判率 三、布隆过滤器的实现四、布隆过滤器的应用场景1. 网络爬虫2. 缓存穿透防护3. 垃圾邮件过滤4. 分…

给stm32cubeide编译出来的bin文件追加crc32

在工程目录下创建ci目录&#xff0c;将AddCrc32.exe丢进去&#xff0c;在stm32cubeide的properties----C/C Build----Settings----Build Steps----Post-build steps Command:添加AddCrc32.exe的路径: source code如下&#xff1a; #include <stdio.h> #include <stdl…

算法-集合的使用

1、set常用操作 set<int> q; //以int型为例 默认按键值升序 set<int,greater<int>> p; //降序排列 int x; q.insert(x); //将x插入q中 q.erase(x); //删除q中的x元素,返回0或1,0表示set中不存在x q.clear(); //清空q q.empty(); //判断q是否为空&a…

网络地址转换

网络地址转换 网络地址转换(Network Address Translation&#xff0c;NAT)的功能是将企业内部自行定义的私有IP地址转换为Internet上可识别的合法IP地址。由于现行IP地址标准--IPv4的限制&#xff0c;Internet面临着IP地址空间短缺的问题&#xff0c;因此从ISP申请并给企业的每…

4.大语言模型预备数学知识

大语言模型预备数学知识 复习一下在大语言模型中用到的矩阵和向量的运算&#xff0c;及概率统计和神经网络中常用概念。 矩阵的运算 矩阵 矩阵加减法 条件&#xff1a;行数列数相同的矩阵才能做矩阵加减法 数值与矩阵的乘除法 矩阵乘法 条件&#xff1a;矩阵A的列数 矩阵…

leetcode hot100刷题日记——35.子集

解答&#xff1a; 方法一&#xff1a;选or不选的dfs&#xff08;输入视角&#xff09; 思路&#xff1a;[1,2,3]的全部子集可以看成是对数组的每一位数字做选择。 eg.空集就是一个数字都不选&#xff0c;[1,2]就是1&#xff0c;2选&#xff0c;3不选。 class Solution { pub…

【数据库】关系数据库标准语言-SQL(金仓)下

4、数据查询 语法&#xff1a; SELECT [ALL | DISTINCT] <目标列表达式> [,<目标列表达式>] … FROM <表名或视图名>[, <表名或视图名> ] … [ WHERE <条件表达式> ] [ GROUP BY <列名1> [ HAVING <条件表达式> ] ] [ ORDER BY <…