Verilog代码。始终声明需要帮助

时间:2012-06-29 19:20:54

标签: verilog

以下是代码: 代码的目的是计算来自寄存器之一的前导零的数量。 我只想从寄存器有效数据中计算一次前导零。我必须使用一个总是块。现在我最初分配了一个1'b1,后来我将它改为1'b0,这样块就会被执行一次。如果我尝试模拟代码。始终块不会执行。但是如果我后来指定为0'b1(这没有任何意义)。代码在模拟器中正确模拟。但是,如果我在FPGA工具包中合成代码,它会产生一些错误的结果。请帮帮我

integer count,index;
wire a;
assign a=1'b1;
always@(a)
begin
    for(count=0;count<7;count=count+1) begin
        index=4*count;
        if((significand[index  ]==1'b0) && (significand[index+1]==1'b0) &&
           (significand[index+2]==1'b0) && (significand[index+3]==1'b0))
             lzero=lzero+1;
    end 
end
assign a=1'b0;
// If I use assign a=0'b1, it simulates properly, 
// but 0'b1 doesn't make any sense, also If I keep 0'b1, 
// I dont get proper result in actual synthesis onto the board.

实际上,我提出问题的意图是,我应该如何使用“始终”块。 因为我只想执行一次这个块,所以我不需要总是设置“posedge clk”或“negedge clock”。所以我该怎么做 ?? 请帮忙,因为我的项目要求我高度使用 循环 if else 循环

3 个答案:

答案 0 :(得分:5)

你看起来像是一个试图编写程序的软件工程师,而不是试图描述硬件。

当你编码时,你必须非常平行地思考。

你的代码说“连续并始终将一个值分配给a”然后它会“连续地并且总是将值0分配给”。这显然不是你想要的。那些赋值语句不是你想要的那些时间。

所以你想要这样的东西(在伪代码中): 1)读取输入寄存器 2)计算0的数量 3)输出0的数量

在软件中你会像这样做第2步:

for i=msb to lsb loop
   if (input[i] == 1) then break;
end loop
return i;

对于HW中的循环,或者意味着编写一个状态机(想想你如何使用函数语言进行递归,其中一个输入是你要做的索引)。或者我们可以自己展开循环。

if (msb == 1 ) then return 0
else if (msb-1 == 1) then return 1
...
else if (msb-31 == 1) then return 31
else return 32
end if

答案 1 :(得分:1)

你可以做的是创建一个标志,在开始时将标志设置为0并在你的始终块中,检查标志的值,如果标志等于0则意味着这是你的第一次标志,执行您的操作并在那里,将您的旗帜设置为1.

这样您就可以保证在启动时只执行一次这些操作。

答案 2 :(得分:1)

这里有一些似乎是错误地使用verilog语言来描述硬件。

如果这是一个需要初始化的测试平台组件,那么initial begin .. end将是实现此目的的方法。

如果这个用于合成并且只运行一次,那么always块不会让我成为设计它的正确方法。如果要在加电时运行一次,则可以预先计算(参数),因此在编译/综合中固定。

如果它基于可以改变的输入,则应该没有启用运行计算。由于此输入可能未在时间0设置,具体取决于硬件的上电顺序。

如果这是用于合成,则以下更合适:

module count_lzero(
  input             clk,
  input             rst_n,
  input             enable, // Toggle high 1 Clk to calc new lzero
  output reg [31:0] lzero
);

always @(posedge clk or negedge rst_n) begin
  if (~rst_n) begin
    lzero <= 'b0;
  end
  else if ( enable ) begin
    for(count=0;count<7;count=count+1) begin
        index=4*count;
        if((significand[index  ]==1'b0) && (significand[index+1]==1'b0) &&
           (significand[index+2]==1'b0) && (significand[index+3]==1'b0))
             lzero<=lzero+1;
    end 
  end
end