# RAM Inferencing in Synplify® Software Using Xilinx RAMs



# **Overview**

Many FPGA families provide a mechanism to implement technology-specific RAMs in HDL source code. To take advantage of these optimal RAM implementations, you must manually instantiate the technology-specific RAM cells.

# **Disadvantages of Instantiation**

The following list outlines the disadvantages of instantiating the technology-specific RAM cells.

- The HDL code is no longer technology independent.
- If you use a black box methodology, your synthesis tool might not have access to any timing or area data.

Synplify software version 7.1 addresses these issues by automatically inferring synchronous RAMs directly from your HDL source code. The RTL View of HDL Analyst® then displays the RAM as a simple component, which makes reading the schematic easier. Additionally, the RAM logic is automatically mapped to applicable technology-specific RAM cells. The Synplify software supports synchronous RAMs for Altera, Atmel, Lattice Orca, and Xilinx technology families. This application note specifically covers the RAM inferencing of Xilinx technology families in the Synplify software.

# **Advantages of Inferencing**

RAM inferencing also has the advantages listed below:

- Technology-independent coding style.
- Synplify software provides automatic timing-driven synthesis for RAMs.
- No additional tool dependencies.

The goal for RAM inferencing in the Synplify software is to give you a method that lets you easily specify RAM structures in your HDL source code, while maintaining portability and ensuring that the netlist output after synthesis remains logically correct. Portability across vendors requires that each vendor technology that is mapped has a certain amount of *glue* logic which normally surrounds the technology-specific RAM primitive so that the logic matches the functionality of the specific RAM module in the Synplify HDL-source RAM primitive. Xilinx-specific details regarding *glue* logic are explained in the "Virtex Conflict Resolution" section. The addition of the glue logic can result in a non-optimal RAM implementation. However, if you want a design that most efficiently uses a specific RAM primitive technology, you must instantiate the vendor-specific RAM primitive.

# Synplify Tool RAM Inferencing Support

To infer a RAM, the Synplify synthesis tool looks for an assignment to a signal (register in Verilog) that is an array of an array, or a case structure controlled by a clock edge and a write enable. If the address used to index the write-to and read-from RAM is the same, then a single-port RAM is inferred as shown in the example below. If the addresses are different, then a dual-port RAM is inferred.

In addition to this support for inferring RAMs, from the Synplify 7.0 software release forward, new support lets you infer Xilinx block SelectRAMs with new coding styles when the RAM output is registered. The new coding style supports the enable and reset (ssrt in the case of Virtex-II) pins of the block SelectRAM primitive. Different write mode operations are supported for single-port RAM targeted for the Virtex-II technology. For more details on these coding styles refer to *Coding Style Mapped to Single-Port Block SelectRAMs* on page 18.

#### VHDL Single-Port RAM Example

The following code illustrates an example of a single-port RAM.

```
library ieee;
use ieee.std logic 1164.all;
use ieee.std logic signed.all;
entity ramtest is
port (q : out std logic vector(3 downto 0);
       d : in std logic vector(3 downto 0);
       addr : in std logic vector(2 downto 0);
                   : in std logic;
       clk : in std logic);
end ramtest;
architecture rtl of ramtest is
type mem type is array (7 downto 0) of std logic vector (3 downto 0);
signal mem : mem type;
begin
q <= mem(conv integer(addr));</pre>
process (clk, we, addr) begin
if rising edge(clk) then
if (we = '1') then
      mem(conv integer(addr)) <= d;</pre>
end if;
end if;
end process;
end rtl;
```



Figure 1: HDL Analyst RTL view of the preceding inferred single-port RAM

#### **Verilog Memory Array**

```
The following code implements a Verilog memory array.

module ramtest(z, raddr, d, waddr, we, clk);

output [3:0] z;

input [3:0] d;

input [3:0] raddr, waddr;

input we;

input clk;

reg [3:0] mem [7:0];

assign z = mem[raddr];

always @(posedge clk) begin

if(we) mem[waddr] = d;

end

endmodule
```

# NOTE: This will implement RAM in the FPGA fabric (called "distributed RAM")



Figure 2: HDL Analyst RTL view of inferred dual -port RAM.

#### Verilog Code Example of a Dual-Port RAM

The following code illustrates an example of a dual-port RAM.

```
module ram16x8(z, raddr, d, waddr, we, clk);
output [7:0] z;
input [7:0] d;
input [3:0] raddr, waddr;
input we;
                                      different read/write addr
input clk;
reg [7:0] z;
reg [7:0] mem0, mem1, mem2, mem3, mem4, mem5, mem6, mem7;
reg [7:0] mem8, mem9, mem10, mem11, mem12, mem13, mem14, mem15;
always @(mem0 or mem1 or mem2 or mem3 or mem4 or mem5 or mem6 or mem7 or
mem8 or mem9 or mem10 or mem11 or mem12 or mem13 or mem14 or mem15 or
raddr)
begin
case (raddr[3:0])
4'b0000: z = mem0;
4'b0001: z = mem1;
                             dual-port RAM allows you to read
4'b0010: z = mem2;
                             from one memory location while
4'b0011: z = mem3;
                             writing to another memory location
4'b0100: z = mem4;
4'b0101: z = mem5;
4'b0110: z = mem6;
4'b0111: z = mem7;
4'b1000: z = mem8;
4'b1001: z = mem9;
4'b1010: z = mem10;
4'b1011: z = mem11;
4'b1100: z = mem12;
4'b1101: z = mem13;
4'b1110: z = mem14;
4'b1111: z = mem15;
```

```
endcase
end
always @(posedge clk) begin
if(we) begin
case (waddr[3:0])
4'b0000: mem0 = d;
4'b0001: mem1 = d;
4'b0010: mem2 = d;
4'b0011: mem3 = d;
4'b0100: mem4 = d;
4'b0101: mem5 = d;
4'b0110: mem6 = d;
4'b0111: mem7 = d;
4'b1000: mem8 = d;
4'b1001: mem9 = d;
4'b1010: mem10 = d;
4'b1011: mem11 = d;
4'b1100: mem12 = d;
4'b1101: mem13 = d;
4'b1110: mem14 = d;
4'b1111: mem15 = d;
endcase
end
end
endmodule
```

# correct figure



Figure 3: HDL Analyst RTL view of the preceding inferred dual-port RAM.

# Inferring Block SelectRAMs in Xilinx

This sections discusses synchronous Xilinx block SelectRAMs and their requirements.

# Fully Synchronous RAMs and Registered Address Requirement

Xilinx block SelectRAMs are fully synchronous. To map to a block SelectRAM, one of the following registered conditions must exist:

- Either the read address or the output must be registered
- Both the read address and the output must be registered

# Using the syn\_ramstyle Attribute for Block SelectRAMs

The syn\_ramstyle="block\_ram" attribute is only required for Xilinx Virtex block SelectRAM and must be set in one of two places to infer block SelectRAMs. You can set the syn\_ramstyle attribute on a memory object in the HDL source code, with TCL script, or the SCOPE® interface as follows:

- In your HDL source code on the register signal used to hold the values of the output of the RAM.
- In Tcl (script)/SCOPE interface(GUI) on the output signal of the RAM.

# **Attribute Usage**

The following examples illustrate how to specify the syn\_ramstyle attribute in various HDL languages. Tcl script, and the SCOPE interface.

#### Verilog Example of Specifying the syn\_ramstyle Attribute

reg [7:0] ram dout [127:0] /\*synthesis syn ramstyle = "block ram"\*/;

#### VHDL Example of Specifying the syn\_ramstyle Attribute

attribute syn\_ramstyle of ram\_dout : signal is "block\_ram";

#### Tcl Example of Specifying the syn\_ramstyle Attribute

define\_attribute { ram\_dout [127:0]} syn\_ramstyle {block\_ram}

#### <unsaved> (constraint File ) \* \_ 🗆 × Enabled Object 1 7 am dout syn\_ramstyle block\_ram 2 $\overline{\mathbf{v}}$ 3 굣 4 $\overline{\mathbf{v}}$ 5 굣 Clock to Clock \ Inputs/Outputs \ Registers \ Multi-Cycle Paths \ Attributes \ Other

#### SCOPE Interface Example of Specifying the syn\_ramstyle Attribute

Figure 4: Setting the syn\_ramstyle attribute in the SCOPE interface to infer block SelectRAMs

# **Global Limitations**

The following global limitations exist when inferencing RAMs:

- RAM inferencing is only supported for synchronous RAMs.
- Initialization of RAMs is not supported
- *Address wrapping* is not supported. This means that the RAM implemented is assumed to start at address 0 and uses one of the following addressing scenarios.

#### Scenario I

The required RAM primitive is 16 words deep and has an address range of 0 to 23 (or 24 words deep). The inferred RAM is implemented in 2 RAM cells, leaving address 24 to 31 unused.

#### Scenario 2

The required RAM primitive is 16 words deep and has an address range of 8 to 23 (or 16 words deep). The inferred RAM is implemented in 2 RAM cells, leaving address 0 to 7, and 24 to 31 unused.

# Implementation Conventions for Specifying Xilinx Block SelectRAMs

The following conventions are used when specifying block SelectRAMs.

Size Requirement: (RAM width > 1 bit) and (RAM depth > 1 bit) and (RAM width \* RAM depth >= 8 bits)

**RAM Primitive:** Use RAM16X1S for single-port RAMs, RAM16X1D for dual-port RAMs.

**Block RAM Primitive:** Use one of the following, based on the technology and word width:

- RAMB4\_S# for single-port block SelectRAMs and RAMB4\_S#\_S# for dual-port block SelectRAMs in Virtex/VirtexE where # is the word width of the RAM.
- RAMB16\_S# for single-port block SelectRAMs, RAMB16S#\_S# for dual-port block SelectRAMs where # is the word width of the RAM.

# Inferring Block SelectRAMs in Xilinx Technologies

RAM inferencing in the Synplify tool is limited to the coding styles discussed throughout this application note.

Prior to the Synplify 7.0 release, a block SelectRAM could be inferred only if the read address was registered as shown by the following code example.

#### Verilog Code Example Inferring Single-Port Block SelectRAM

```
module ram test(q, a, d, we, clk);
output [7:0] q;
input [7:0] d;
input [6:0] a;
input clk, we;
reg [6:0] read add;
/* The array of an array register ("mem") the RAM will be inferred from.
reg [7:0] mem [127:0] /* synthesis syn ramstyle = "block ram */;
assign q = mem[read add];
always @(posedge clk) begin
                                     makes implementation in BRAM
if(we)
/* Register RAM Data */
mem[a] <= d;
/* Register Read Address. Basic RAM support does not
require this address register.*/
read add <= a;
end
endmodule
```

# **Dual-Port Block SelectRAM with Registered Read Address**

When two addresses are used to do the read and the write operation respectively, and the read address is registered, a dual-port block SelectRAM can be inferred as shown by the following example and illustrated in *HDL Analyst Technology view of an inferred Virtex block SelectRAM* on page 9.

#### **Dual-Port Block SelectRAM with Read Address Registered**

```
module dualportram(q, a1, a2, d, we, clk, en);
output [7:0] q;
input [7:0] d;
input[6:0] a1;
input[6:0] a2;
input clk, we, en;

reg [6:0] read_addr;
reg[7:0] mem [127:0] /* synthesis syn_ramstyle="block_ram" */;
```

```
assign q = mem[read_addr];
always @(posedge clk) begin
if (we)
mem[a2] <= d;
read_addr <= a1;
different addr
end</pre>
```

endmodule



Figure 5: HDL Analyst Technology view of an inferred Virtex block SelectRAM

This figure shows a dual-port RAM inferred by the code in the preceding example, *Dual-Port Block SelectRAM with Registered Read Address* on page 8.