Verilog 向量 Packing/Unpacking 宏

Verilog Vector Packing/Unpacking Macro

我目前正在努力解决 verilog 模块仅接受一维压缩向量作为 inputs/outputs 的问题。例如:

wire [bitWidth-1:0] data;

我想要做的是输入一个一维压缩的二维向量(想想数字数组)。例如:

wire [bitWidth-1:0] data [0:numData-1];

我目前有几个方便的宏,可以展平二维向量和展平一维向量。他们是:

`define PACK_ARRAY(PK_WIDTH,PK_LEN,PK_SRC,PK_DEST) \
genvar pk_idx; \
generate \
    for (pk_idx=0; pk_idx<(PK_LEN); pk_idx=pk_idx+1) begin : packLoop \
        assign PK_DEST[((PK_WIDTH)*pk_idx+((PK_WIDTH)-1)):((PK_WIDTH)*pk_idx)] = PK_SRC[pk_idx][((PK_WIDTH)-1):0]; \
    end \
endgenerate

`define UNPACK_ARRAY(PK_WIDTH,PK_LEN,PK_DEST,PK_SRC) \
genvar unpk_idx; \
generate \
    for (unpk_idx=0; unpk_idx<(PK_LEN); unpk_idx=unpk_idx+1) begin : unpackLoop \
        assign PK_DEST[unpk_idx][((PK_WIDTH)-1):0] = PK_SRC[((PK_WIDTH)*unpk_idx+(PK_WIDTH-1)):((PK_WIDTH)*unpk_idx)]; \
    end \
endgenerate

如果我只使用它们来处理我正在使用的模块的一个输出和一个输入,那么这些宏非常有用。问题是,如果我在给定模块中多次使用这些宏中的任何一个,那么出现以下错误:

1) genvar 已经声明。 2) 循环标签已经被使用

我可以通过不在宏中声明它并在其他地方声明它来解决 genvar 问题,但我不知道如何解决循环标签问题并且仍然能够使用宏保持干净的代码。

欢迎提出任何建议('switch to system verilog' :P 除外)!谢谢。

对于纯 verilog 解决方案,您需要为 genvar 标识符添加一个参数。示例(注意:为简单起见,将 ((PK_WIDTH)*unpk_idx+(PK_WIDTH-1)):((PK_WIDTH)*unpk_idx) 替换为 (PK_WIDTH)*unpk_idx +: PK_WIDTH,请参阅 here & here):

`define UNPACK_ARRAY(PK_WIDTH,PK_LEN,PK_DEST,PK_SRC,UNPK_IDX) \
genvar UNPK_IDX; \
generate \
  for (UNPK_IDX=0; UNPK_IDX<(PK_LEN); UNPK_IDX=UNPK_IDX+1) begin //: unpackLoop <-- remove label\
    assign PK_DEST[UNPK_IDX][((PK_WIDTH)-1):0] = PK_SRC[(PK_WIDTH)*unpk_idx +: PK_WIDTH]; \
  end \
endgenerate

如果 SystemVerilog 是使用流运算符的选项,则有一个更简单的解决方案。请参阅 IEEE Std 1800-2012 § 11.4.14 流式处理运算符 (pack/unpack)。示例:

// packing
assign packed_array = {>>{unpacked_array}};
// unpacking
assign unpacked_array = {<<PK_WIDTH{packed_array}};