FPGA 组合逻辑和时序逻辑

vivado在进行实现阶段的opt_design过程报错,出现多驱动问题;

DRC MDRV-1\] multiple Drive Nets:Net UNCONN_OUT\[1\] has multiple drivers: genblk1\[0\].tmp_max_1d_reg\[1\]\[1\]/Q,and genblk1\[1\].tmp_max_1d_reg\[1\]\[1\]/Q; 问题代码1: 'd1: begin // 实现每2个数累加求和,320个数,一共输出160个 group_size = 2; group_num = 320 / group_size; // 160组 group_idx = m; // 每组输出一个累加和 if(group_idx < group_num) begin start_idx <= group_idx * group_size; temp_sum <= 'd0; // 计算组内累加和 for(i=0; i max_val) begin max_val <= din[start_idx+i]; end if(din[start_idx+i] < min_val) begin min_val <= din[start_idx+i]; end end // 分配输出 if(m % 2 == 0) begin // 偶数位置输出最大值 dout[m] <= max_val; end else begin // 奇数位置输出最小值 dout[m] <= min_val; end end else begin dout[m] <= 'd0; end end 分析原因,对于赋值,尽管这些逻辑是在一个always 块里面实现的,但是在for 循环中进行循环计算再赋值,不应该用非阻塞赋值,而是要用阻塞赋值; 正确代码1: 'd1: begin // 实现每2个数累加求和,320个数,一共输出160个 group_size = 2; group_num = 320 / group_size; // 160组 group_idx = m; // 每组输出一个累加和 if(group_idx < group_num) begin start_idx <= group_idx * group_size; temp_sum <= 'd0; // 计算组内累加和 for(i=0; i max_val) begin max_val = din\[start_idx+i\]; end if(din\[start_idx+i\] \< min_val) begin min_val = din\[start_idx+i\]; end end // 分配输出 if(m % 2 == 0) begin // 偶数位置输出最大值 dout\[m\] \<= max_val; end else begin // 奇数位置输出最小值 dout\[m\] \<= min_val; end end else begin dout\[m\] \<= 'd0; end end 区别分析: | 特性 | 第一段代码(正常) | 第二段代码(有多驱动) | |----------|------------------------------------|---------------------------------------------| | **赋值方式** | 阻塞赋值 (`=`) | 非阻塞赋值 (`<=`) | | **变量性质** | 临时变量(组合逻辑) | 推断为寄存器(时序逻辑) | | **执行时机** | 立即计算,在同一时钟周期内完成 | 值在时钟边沿后更新,延迟生效 | | **关键影响** | 每个 `m` 实例独立计算自己的 `max_val/min_val` | **所有 `m` 实例共享并竞争同一个 `max_val/min_val` 寄存器** | **1. 非阻塞赋值 (`<=`) 的硬件含义** 在 `always @(posedge i_clk)` 块中使用 `<=` 赋值,综合工具会**推断出寄存器**。这意味着: * `max_val` 和 `min_val` 不再是临时计算变量,而是会在时钟沿之间保持状态的**物理寄存器**。 * 循环中的 `max_val <= din[start_idx+i];` 不是立即更新,而是安排一个"计划",在时钟周期结束时更新。因此,`for` 循环中后续的比较使用的 `max_val` 值始终是**该寄存器在上一个时钟周期的旧值**,这完全打破了查找最大值的逻辑。 **2. 多驱动冲突的产生** 这是最关键的一点:**在 `generate` 循环中,每个展开的 `m` 实例都包含自己的 `always` 块,但所有这些块都试图驱动名为 `max_val` 和 `min_val` 的同一个(全局)寄存器。** * 当 `group_idx < 2` 时(即 `m = 0, 1, 2, 3`),**4个并行的硬件实例**会在每个时钟沿同时执行。 * 每个实例都试图根据自己的 `start_idx` 和循环逻辑,向 `max_val` 和 `min_val` 写入数据。 * 这就造成了4个驱动源同时驱动同一根线,即 **`multiple driver`** 错误。 **第一段代码之所以正确** ,是因为它使用了阻塞赋值 (`=`),且**没有将 `max_val/min_val` 声明为 `reg`** (它们很可能是 `integer` 或隐式的临时变量)。这样,每个 `m` 实例的 `always` 块中的 `max_val` 和 `min_val` 都是该实例**本地、独立的组合逻辑变量**,彼此互不干扰。 在时序逻辑的 `always` 块中,`<=` 用于推断寄存器,`=` 用于临时计算。 **设计原则** :**"计算用阻塞,输出用非阻塞"**。在时钟触发的块内进行复杂的组合逻辑计算时,使用阻塞赋值和局部变量;只有需要寄存到下一个时钟周期的最终结果,才使用非阻塞赋值输出。

相关推荐
ZPC82105 天前
docker 镜像备份
人工智能·算法·fpga开发·机器人
ZPC82105 天前
docker 使用GUI ROS2
人工智能·算法·fpga开发·机器人
tiantianuser5 天前
RDMA设计53:构建RoCE v2 高速数据传输系统板级测试平台2
fpga开发·rdma·高速传输·cmac·roce v2
博览鸿蒙5 天前
FPGA 和 IC,哪个前景更好?怎么选?
fpga开发
FPGA_小田老师5 天前
xilinx原语:ISERDESE2原语详解(串并转换器)
fpga开发·iserdese2·原语·串并转换
tiantianuser6 天前
RDMA设计50: 如何验证网络嗅探功能?
网络·fpga开发·rdma·高速传输·cmac·roce v2
Lzy金壳bing6 天前
基于Vivado平台对Xilinx-7K325t FPGA芯片进行程序在线更新升级
fpga开发·vivado·xilinx
unicrom_深圳市由你创科技6 天前
医疗设备专用图像处理板卡定制
图像处理·人工智能·fpga开发
tiantianuser6 天前
RDMA设计52:构建RoCE v2 高速数据传输系统板级测试平台
fpga开发·rdma·高速传输·cmac·roce v2
luoganttcc6 天前
Taalas 将人工智能模型蚀刻到晶体管上,以提升推理能力
人工智能·fpga开发