2026/3/19 15:04:40
网站建设
项目流程
FPGA 设计与处理器程序设计核心差异在于:处理器是指令顺序执行,而 FPGA 以晶振产生的周期性时钟为核心驱动,通过匹配不同时钟周期完成计数,再依据计数值执行各类动作,最终实现特定功能;计数器是 FPGA 时序设计中最基础、最核心的模块,其设计核心围绕时钟周期计算、条件优先级划分、标准时序逻辑编写展开,具体使用方法总结如下:
一、先做时钟周期精准计算,确定计数阈值
- 由晶振频率算出单个时钟周期(周期 = 1 / 频率),如 100MHz 时钟的单周期为 10ns;
- 根据实际需求(如定时 1 秒、分频到 1Hz),计算所需总时钟周期数,总周期数 = 目标时间 / 单时钟周期;
- 确定计数阈值:因计数器从 0 开始累加,最终计数阈值为总周期数 - 1,如 100MHz 时钟定时 1 秒需 100,000,000 个周期,计数阈值则为 99,999,999。
二、拆解计数器三大核心条件,明确优先级排序
计数器设计需拆解为 3 类核心条件,且按 “高优先级到低优先级” 依次处理,优先级排序为:复位条件(最高)→ 清零条件(次之)→ 累加条件(最低)
- 复位条件:上电初始化或异常恢复时触发,用于将计数值置初始值(通常为 0),保证计数器上电状态可知,分为异步复位(无需时钟沿,复位信号有效立即执行)、同步复位(需时钟沿触发才执行),工程中异步复位为常用方案;
- 清零条件:计数值达到预设阈值后触发,将计数器置 0 以实现循环计数,若不优先处理清零,计数值会溢出超出预设范围,导致清零条件失效;
- 累加条件:计数器的正常工作条件,由时钟沿驱动,在无复位、无清零的情况下,每个时钟沿计数值加 1,是计数器实现周期计数的基础。
三、编写标准时序逻辑代码,遵循 FPGA 编码规范
以最常用的异步复位计数器为例,采用单一时序 always 块实现(一个时钟域 + 一组关联时序动作,符合 FPGA 最佳编码实践),核心逻辑严格遵循 “优先级从高到低”,代码框架与规范如下:
- 触发列表:异步复位需包含时钟上升沿 + 复位下降沿(
posedge 时钟名 or negedge 复位名),同步复位仅需时钟上升沿; - 赋值规则:时序逻辑必须使用非阻塞赋值
<=,模拟硬件寄存器的并行更新特性; - 代码结构:按 “复位→清零→累加” 的优先级依次编写,无冗余分支,保证综合器可生成高效、正确的硬件电路。
标准代码模板(以 100MHz 时钟、1 秒定时为例):
verilog
// 定义计数器寄存器(根据计数阈值确定位宽,32位可满足绝大多数低频定时需求) reg [31:0] cnt; // 1秒定时计数器 // 异步复位计数器核心时序逻辑 always @(posedge SYS_CLK or negedge RST_N) begin // SYS_CLK:系统时钟,RST_N:低有效异步复位 if(!RST_N) begin // 1. 最高优先级:复位条件,计数值置0 cnt <= 32'd0; end else if(cnt == 32'd99_999_999) begin // 2. 次之:清零条件,计满1秒置0 cnt <= 32'd0; end else begin // 3. 最低优先级:累加条件,每个时钟沿计数值+1 cnt <= cnt + 1'd1; end end
四、核心设计要点
- 区分软硬件设计思维:无需像处理器程序那样考虑执行顺序,只需明确硬件条件的优先级,先处理特殊情况(复位、清零),再处理正常工作情况(累加),这是 FPGA 计数器设计的核心思路;
- 禁止优先级写反:若将 “累加” 写在 “清零” 前,计数器计满后会先累加再清零,导致计数值溢出,清零条件永久失效,计数器无法循环工作;
- 寄存器必做复位:所有计数器寄存器必须设置复位值,避免 FPGA 上电后计数器状态未知,导致整个时序逻辑功能异常;
- 单 always 块设计:同一时钟域、同一复位方式的计数器,需在单个 always 块内完成所有逻辑编写,无需按信号拆分,既简化代码,又利于综合器优化硬件电路。