

POLITECNICO DI TORINO

# ELECTRONIC ENGINEERING EMBEDDED SYSTEMS

# Microelectronic Systems Read-only and Read-write Memories

Authors:
Andrea PATERNÒ 206526
Giuseppe Luca MACALUSO 206009

November 27, 2014

## 1 Introduction

One of the main features in Harvard architectures is the net separation of the Instruction Ram and the Data Ram, along with their buses. The Instruction Ram is usually a static Read-Only Memory, while the Data Ram a true Random Access Memory. Therefore the following implementation works in this direction, providing two different components, one to be used for the intruction management, and one for the data.

However, their management is quite similar.

# 2 Read-only RAM

The read-only ram works in two different ways, depending on the value of the **RST** signal.

The ROM entity is listed below, and is to be used as a reference of this chapter's signal names.

Listing 1: Read Only Memory entity

```
entity ROMEM is
    generic (
         FILE PATH
                       : string;
                                                    -- ROM data file
                                                    -- Number of lines in the ROM
-- Number of bits per word
         ENTRIES
                       : integer := 128;
         WORD SIZE
                       : integer := 32;
         \overline{DATA} \ \overline{DELAY} : natural := 2
                                                    - Delay ( in # of clock cycles )
    );
    port (
         CLK
                                 : in std_logic;
         RST
                                 : in std_logic;
                                 : in std_logic_vector(WORD_SIZE - 1 downto 0);
: in std_logic;
         ADDRESS
         ENABLE
         DATA READY
                                : out std_logic;
                                 : out std logic vector (2*WORD SIZE - 1 downto 0)
         DATA
end ROMEM;
```

# 2.1 Reset high

If **RST** is high, than the ROM refreshes its internal buffer contents with the ones in the dump file, whose path is to be set in the **FILE\_PATH** variable. The data to be provided must be in hexadecimal form, just as the compilation dump of the compiler provided. The **ENTRIES** variable holds the number of instructions ( data words ) that the ROM can hold.

The WORD\_SIZE variable is already set to 32, so as to be compliant with the compiler itself, which encodes the ASM instruction is a classical MIPS 32-bit fixed-length encoding.

#### 2.2 Reset low

If **RST** is low, than depending on the value of the **ENABLE** signal, the ROM is either ready to accept incoming requests ( **ENABLE** high ) or simply remains in its idle state ( **ENABLE** low ).

The **DATA\_DELAY** variable is used to simulate the delay in real-world memories, and is expressed in numbers of clock cycles.

**Usage** In order to send a request to the memory, we must drive the signals as follows:

- CLK -> Clock of the memory. Can be equal to the one used by the uP.
- RST  $\rightarrow$  Low
- ENABLE -> High
- ADDRESS -> The address is a vector of **WORD\_SIZE** bits, representing the index of the word to be read. It has to be stable until the **DATA\_READY** signal becomes high.

After **DATA\_DELAY** clock cycles, the ROM will drive **DATA\_READY** high and place the data in the bus named **DATA**.

## 3 Read-write RAM

Many of the signals in the RAM entity maintain the same name and functionality as the ones in the ROM. Also its management works in the same way, but for a few changes detailed below.

Listing 2: Read Write Memory entity

```
entity RWMEM is
    generic (
             FILE_PATH: string;
FILE_PATH_INIT: string;
                                                 -- RAM output data file
                                                 -- RAM initialization data file
             WORD SIZE: natural := 32;
                                                 -- Number of bits per word
             ENTRIËS:
                                                 -- Number of lines in the ROM
                         natural := 128;
             DATA DELAY: natural := 2
                                                 -- Delay ( in # of clock cycles )
    port (
                                    : in std_logic;
: in std_logic;
             CLK
             RST
             ADDRESS
                                    : in std_logic_vector(WORD_SIZE - 1 downto 0);
                                   : in std_logic;
: in std_logic;
             ENABLE
             READNOTWRITE
             DATA_READY
                                   : out std_logic;
```

```
\label{eq:inout_def} $\operatorname{INOUT\_DATA}$ : $\operatorname{inout}$ std_logic_vector(2*WORD_SIZE-1 $\operatorname{downto}$ 0)$ ; end $\operatorname{RWMEM}$; }
```

The **DATA** bus in the ROM component was replaced by the **INOUT\_DATA** bus, which (guess?) this time is driven by tri-state buffers, as, depending on the mode of operation, holds either the data to be written in the RAM, or read from it.

The type of operation to be performed in controlled by the **READ-NOTWRITE** signal, which, if high, tells the RAM that we're interested in reading the memory cell ( and its adjacent one, in this current implementation ) located at ÆDDRESS; while if low, that we're interested in writing to those cells with the contents of the bus.

In order to reduce the number of pins of the component, in fact, we chose to use a shared buffer for both the operations, excluding the case in which there is a concurrent writing and reading.

## 4 The testbench

The testbench only instantiates the DLX, RAM and ROM components and connects them together. It also generates the global clock and reset signals.

Listing 3: Component Instantiation

```
IRAM
IRAM : ROMEM
    generic map ("/home/gandalf/Desktop/dlx/rocache/hex.txt")
    port map (CLK, RST, IRAM_ADDRESS, IRAM_ENABLE, IRAM_READY, IRAM_DATA);
 - DRAM
D\!R\!A\!M\ :\ R\!W\!M\!E\!M
    generic map (
        "/home/gandalf/Desktop/dlx/rwcache/hex_init.txt",
        "/home/gandalf/Desktop/dlx/rwcache/hex.txt"
    port map ( CLK, RST, DRAM ADDRESS, DRAM ENABLE,
        DRAM READNOTWRITE, DRAM READY, DRAM DATA
  - DLX
GIANLUCA : DLX
    port map ( CLK, RST, IRAM_ADDRESS, IRAM_ENABLE, IRAM_READY,
        IRAM DATA, DRAM ADDRESS, DRAM ENABLE, DRAM READNOTWRITE,
        DRAM_READY, DRAM_DATA
    );
```