Lattice Diamond 如何将初始 RAM 值映射到 EBR 原语?

How does Lattice Diamond map initial RAM values to the EBR primitives?

我正在从磁盘读取 4 KiB(1Ki x 32 位)真双端口 RAM 的初始 RAM 值。因为我的莱迪思 ECP5 设备只有 18 位宽的嵌入式块 RAM (EBR),所以将 2 个 RAM 组合起来以匹配 32 位的字长。

我的 ocram.mem 文件有 4 行数据,每行有 8 个十六进制数字(=> 32 位字行)。
该文件如下所示:

@000
44332211
88776655
CCBBAA99
00FFEEDD

我的读取函数在 LSE 日志中报告了这些行:

INFO - synthesis: d:/git/poc/src/mem/mem.pkg.vhdl(144): Note: "44332211". VHDL-1533
INFO - synthesis: d:/git/poc/src/mem/mem.pkg.vhdl(144): Note: "88776655". VHDL-1533
INFO - synthesis: d:/git/poc/src/mem/mem.pkg.vhdl(144): Note: "CCBBAA99". VHDL-1533
INFO - synthesis: d:/git/poc/src/mem/mem.pkg.vhdl(144): Note: "00FFEEDD". VHDL-1533

所以,第一行被按预期跳过,后面的行字符串也没有问题。行读取和 char 到 std_logic_vector 转换后,我再次将 2D STD_LOGIC 矩阵输出到 LSE 日志中:

INFO - synthesis: d:/git/poc/src/mem/ocram/ocram_tdp.vhdl(123): Note: "Memory: => 
0100 0100 0011 0011 0010 0010 0001 0001
1000 1000 0111 0111 0110 0110 0101 0101
1100 1100 1011 1011 1010 1010 1001 1001
0000 0000 1111 1111 1110 1110 1101 1101
0000 0000 0000 0000 0000 0000 0000 0000
0000 0000 0000 0000 0000 0000 0000 0000
0000 0000 0000 0000 0000 0000 0000 0000
0000 0000 0000 0000 0000 0000 00   ....   000
0000 0000 0000 0000 0000 0000 0000 0000
0000 0000 0000 0000 0000 0000 0000 0000
0000 0000 0000 0000 0000 0000 0000 0000
0000 0000 0000 0000 0000 0000 0000 0000
". VHDL-1533

可以看出,这些值仍然没有受到影响。
合成无误完成。

所以我开始在 Lattice Diamond 中制作地图和 P&R。平面图向我展示了 2 个使用过的嵌入式块 RAM——这是有意的。但是当我点击 RAM 时,INIT 值看起来很奇怪。

RAM0 INITVAL_00 = 0x0000...000000000000003FF332EE221DD110CC
RAM0 INITVAL_01 = 0x0000...00000000000000000000000000000000
RAM0 INITVAL_.. = 0x.......................................
RAM0 INITVAL_3F = 0x0000...00000000000000000000000000000000

RAM1 INITVAL_00 = 0x0000...00000000000002EDD02A990265502211
RAM1 INITVAL_01 = 0x0000...00000000000000000000000000000000
RAM1 INITVAL_.. = 0x.......................................
RAM1 INITVAL_3F = 0x0000...00000000000000000000000000000000

我承认我的一些价值观,但不是全部。我没有在 EBR 或 ECP5 文档中找到任何线索。

我找到了 RAM 映射模式...

来自 Lattice Diamond Floorplaner 的值:

RAM 0
.....    ..................
00000 => 000000000000000000
02EDD => 000010111011011101
02A99 => 000010101010011001
02655 => 000010011001010101
02211 => 000010001000010001

RAM 1
.....    ..................
00000 => 000000000000000000
003FF => 000000001111111111
332EE => 110011001011101110
221DD => 100010000111011101
110CC => 010001000011001100

当我把这两部分结合起来,引入新的分隔符,就可以解码成原来的内容了:

RAM 1 RAM 0          RAM 1 (17:0)             RAM 0 (13:0)      original
00000 00000 => 0000.0000.0000.0000.00 --.--00.0000.0000.0000 ~> 0000-0000
003FF 02EDD => 0000.0000.1111.1111.11 --.--10.1110.1101.1101 ~> 00FF-EEDD
332EE 02A99 => 1100.1100.1011.1011.10 --.--10.1010.1001.1001 ~> CCDD-AA99
221DD 02655 => 1000.1000.0111.0111.01 --.--10.0110.0101.0101 ~> 8877-6655
110CC 02211 => 0100.0100.0011.0011.00 --.--10.0010.0001.0001 ~> 4433-2211

RAM组织如下:

  • 内存行数:64
  • 每行字数:64
  • 每个字的位数:18(编码为 5 个十六进制数字)

DataIn/DataOut 到内部 RAM 映射:

ram0(13 downto 0) <= DataIn(13 downto 0);
ram1(17 downto 0) <= DataIn(31 downto 14);

DataOut(13 downto 0)  <= rom0(13 downto 0);
DataOut(31 downto 14) <= rom1(17 downto 0);