



# CprE 381: Computer Organization and Assembly Level Programming

Lab Week 4 VHDL

Henry Duwe  
Electrical and Computer Engineering  
Iowa State University



# Modeling Memory in VHDL

```
architecture rtl of mem is

    -- Build a 2-D array type for the RAM
    subtype word_t is std_logic_vector((DATA_WIDTH-1) downto 0);
    type memory_t is array(2**ADDR_WIDTH-1 downto 0) of word_t;

    -- Declare the RAM signal and specify a default value. Quartus Prime
    -- will load the provided memory initialization file (.mif).
    signal ram : memory_t;

begin

    process(clk)
    begin
        if(rising_edge(clk)) then
            if(we = '1') then
                ram(to_integer(unsigned(addr))) <= data;
            end if;
        end if;
    end process;

    q <= ram(to_integer(unsigned(addr)));

end rtl;
```



# Modeling Memory in VHDL

```
architecture rtl of mem is

    -- Build a 2-D array type for the RAM
    subtype word_t is std_logic_vector((DATA_WIDTH-1) downto 0);
    type memory_t is array(2**ADDR_WIDTH-1 downto 0) of word_t;

    -- Declare the RAM signal and specify a default value. Quartus Prime
    -- will load the provided memory initialization file (.mif).
    signal ram : memory_t;

begin

    process(clk)
    begin
        if(rising_edge(clk)) then
            if(we = '1') then
                ram(to_integer(unsigned(addr))) <= data;
            end if;
        end if;
    end process;

    q <= ram(to_integer(unsigned(addr)));

end rtl;
```

Defining data type

Instantiation



# Modeling Memory in VHDL

```
architecture rtl of mem is

    -- Build a 2-D array type for the RAM
    subtype word_t is std_logic_vector((DATA_WIDTH-1) downto 0);
    type memory_t is array(2**ADDR_WIDTH-1 downto 0) of word_t;

    -- Declare the RAM signal and specify a default value. Quartus Prime
    -- will load the provided memory initialization file (.mif).
    signal ram : memory_t;

begin

    process(clk)
    begin
        if(rising_edge(clk)) then
            if(we = '1') then
                ram(to_integer(unsigned(addr))) <= data;
            end if;
        end if;
    end process;

    q <- ram(to_integer(unsigned(addr)));

end rtl;
```

Synchronous write

Asynchronous read



# Word Vs. Byte Addressable memory

## Word Addressable

- This is the format provided in lab
- Must read/write 4 bytes at a time
- Only 4 addresses (0x00-0x03)
- Big Endian format

|    |            |
|----|------------|
|    |            |
| 00 | 0x14120900 |
| 01 | 0x44220215 |
| 10 | 0xFFFFFFFF |
| 11 | 0x00000000 |

## Byte Addressable

- This is the format RISC-V uses
- Can read/write to individual bits
- 16 addresses (0x00-0x0F)

|    |      |      |      |      |
|----|------|------|------|------|
|    | 00   | 01   | 10   | 11   |
| 00 | 0x00 | 0x09 | 0x12 | 0x14 |
| 01 | 0x15 | 0x02 | 0x22 | 0x44 |
| 10 | 0xFF | 0xFF | 0xFF | 0xFF |
| 11 | 0x00 | 0x00 | 0x00 | 0x00 |



# Simulation Hierarchy

```
entity tb_dffg is
| generic(gCLK_PERIOD : time := 50 ns);
end tb_dffg;
```

Simulated Entity

```
DUT: dffg
port map(i_CLK => s_CLK,
          i_RST => s_RST,
          i_WE  => s_WE,
          i_D   => s_D,
          o_Q   => s_Q);
```

```
architecture mixed of dffg is
| signal s_D    : std_logic;    -- Multiplexed input to the FF
| signal s_Q    : std_logic;    -- Output of the FF
```

# Simulation Hierarchy



Simulated Entity



# Simulation Hierarchy

```
entity tb_dffg is
| generic(gCLK_HPER : time := 50 ns);
end tb_dffg;
```

Simulated Entity

```
DUT: dffg
port map(i_CLK => s_CLK,
          i_RST => s_RST,
          i_WE  => s_WE,
          i_D   => s_D,
          o_Q   => s_Q);
```

Component Instantiation

```
architecture mixed of dffg is
| signal s_D    : std_logic;    -- Multiplexed input to the FF
| signal s_Q    : std_logic;    -- Output of the FF
```

# Simulation Hierarchy



Component Instantiation

Simulated Entity



# Simulation Hierarchy

```
entity tb_dffg is
| generic(gCLK_HPER : time := 50 ns);
end tb_dffg;
```

Simulated Entity

```
DUT: dffg
port map(i_CLK => s_CLK,
         i_RST => s_RST,
         i_WE  => s_WE,
         i_D   => s_D,
         o_Q   => s_Q);
```

Component Instantiation

```
architecture mixed of dffg is
| signal s_D    : std_logic; -- Multiplexed input to the FF
| signal s_Q    : std_logic; -- Output of the FF
```

Internal Signals

# Simulation Hierarchy

Internal signals



Component Instantiation

Internal Ports

Simulated Entity

- You can find the address of a component by following this hierarchy
- Ex, the address of the internal signal s\_Q is “/tb\_dffg/DUT/s\_Q”



# Initializing memory

The screenshot shows the ModelSim simulation environment. On the left, the 'sim - Default' window displays a hierarchical tree of design elements. At the top level is 'tb\_mem'. Under 'tb\_mem' are 'DUT' and several other components like 'line\_39', 'line\_48', 'P\_CLK', 'P\_TB', and packages from 'standard', 'textio', and 'std\_logic\_1164'. The 'DUT' node is expanded, showing its internal structure. On the right, the 'Objects' window lists simulation objects with their current values. The 'ram' object is highlighted with a red box, showing its value as '{32'hXXXXXXXX} {...}'. Other objects listed include 'we' (0), 'q' (32'hXXXXXXXX), 'DATA\_WIDTH' (32'h20), 'data' (32'h00000000), 'clk' (0), 'ADDR\_WIDTH' (32'hA), and 'addr' (10'h000). The 'Value' column shows the current value of each signal.

| Name       | Value                | Type    | Direction |
|------------|----------------------|---------|-----------|
| we         | 0                    | Signal  | In        |
| ram        | {32'hXXXXXXXX} {...} | Signal  | Internal  |
| q          | 32'hXXXXXXXX         | Signal  | Out       |
| DATA_WIDTH | 32'h20               | Gene... | In        |
| data       | 32'h00000000         | Signal  | In        |
| clk        | 0                    | Signal  | In        |
| ADDR_WIDTH | 32'hA                | Gene... | In        |
| addr       | 10'h000              | Signal  | In        |

mem load –infile {path to memory file} – format hex {path in simulated hierarchy}

- Our goal in this case is to initialize the RAM signal, so we should use the address
  - /tb\_mem/DUT/RAM