# Relazione Progetto Reti Logiche

Francesco Spangaro - Luca Tosetti April 2023

Matricola: 955106 - 956958 Codice persona: 10734844 - 10739865 Docente: Gianluca Palermo



# 1 Introduzione

# 1.1 Specifiche generali

#### 1.1.1 Descrizione

Il progetto consiste nell'andare a descrivere un modulo HW che si interfacci ad una memoria, estraendone dati sulla base di indirizzi letti da input dal modulo stesso e caricando tali dati su una delle apposite 4 uscite, gestendo la visione dei dati tramite un segnale "DONE".

### 1.1.2 Funzionamento

Più in dettaglio: il funzionamento del modulo prevede di avere inizialmente tutte le uscite e il segnale "DONE" a 0. Questa situazione rimane tale finchè il segnale in ingresso "START" rimane a 0, nel momento in cui il segnale "START" passa a 1, si possono iniziare a leggere i bit presenti sul secondo segnale (seriale) in ingresso "W". La lettura di questi bit continua fintanto che "START" rimane pari a 1, in particolare:

- I primi due bit letti compongono l'indirizzo di una delle 4 uscite, su cui verrà inserito il dato letto da memoria.
- I restanti bit (da un minimo di 0, ad un massimo di 16) verranno usati come indirizzo per accedere alla memoria e leggere il dato presente in tale indirizzo.

Da notare come il segnale "START" rimane pari a 1 per un minimo di 2 cicli di clock (2 bit) fino ad un massimo pari a 18 cicli di clock (18 bit).

A questo punto l'indirizzo letto può essere usato per leggere il dato in memoria, una volta fatto ciò tale dato deve essere caricato sull'uscita indicata dai primi due bit letti dal segnale in ingresso "W".

Ciononostante, il valore delle uscite (compresa quella su cui è stato caricato il dato) deve rimanere nullo fintanto che "**DONE**" rimane pari a 0. Quando "**DONE**" viene settato a 1 (entro 20 cicli di clock dal passaggio di "START" da 1 a 0), il valore caricato sulle uscite deve essere mostrato (anche quelli caricati su altre uscite in iterazioni precedenti) per un solo ciclo di clock, dopodichè il segnale "**DONE**" torna a 0, così come le uscite.

## 1.2 Strumenti forniti

Tra gli strumenti fornitoci, possiamo trovare:

• Interfaccia di memoria:

```
-- Single-Port Block RAM Write-First Mode (recommended template)
-- File: rams_02.vhd
library ieee;
use ieee.std logic 1164.all;
use ieee.std logic unsigned.all;
entity rams_sp_wf is
port(
 clk : in std_logic;
 we : in std_logic;
 en : in std logic;
 addr : in std logic vector(15 downto 0);
 di : in std logic vector(7 downto 0);
 do : out std logic vector(7 downto 0)
);
end rams_sp_wf;
architecture syn of rams_sp_wf is
type ram type is array (65535 downto 0) of std logic vector(7 downto 0);
signal RAM : ram type;
begin
 process(clk)
   begin
   if clk'event and clk = '1' then
     if en = '1' then
       if we = '1' then
         RAM(conv integer(addr)) <= di;
                                 <= di after 2 ns;
       else
         do <= RAM(conv integer(addr)) after 2 ns;
       end if;
     end if;
   end if;
 end process;
end syn;
```

#### • Interfaccia modulo HW:

```
entity project_reti_logiche is
   port (
       i_clk
               : in std_logic;
       i_rst
              : in std_logic;
       i_start : in std_logic;
               : in std_logic;
               : out std_logic_vector(7 downto 0);
       o z0
               : out std_logic_vector(7 downto 0);
       o_z1
       o_z2
               : out std_logic_vector(7 downto 0);
               : out std_logic_vector(7 downto 0);
       o_z3
       o_done : out std_logic;
       o_mem_addr : out std_logic_vector(15 downto 0);
       i_mem_data : in std_logic_vector(7 downto 0);
       o_mem_we : out std_logic;
                 : out std_logic
       o mem en
end project_reti_logiche;
```

#### Più dettagliatamente:

- 1. i clk: segnale di "CLOCK" generato dal testbench
- 2. i\_rst: segnale di "RESET" asincrono generato dal testbench, usato per inizializzare la macchina, pronta per ricevere il primo segnale di "START".
- 3. i start: segnale di "START" generato dal testbench
- 4. i w: segnale di input generato dal testbench
- 5. o z0: primo canale d'uscita
- 6. o z1: secondo canale d'uscita
- 7. o z2: terzo canale d'uscita
- 8. o z3: quarto canale d'uscita
- 9. o done: segnale di uscita che comunica la fine dell'elaborazione
- 10. o\_mem\_addr: segnale che permette di inviare l'indirizzo dal quale leggere il dato alla memoria
- 11. i\_mem\_data: segnale dal quale si riceve il dato letto dalla memoria
- 12. o\_mem\_en: segnale di "ENABLE", necessario per poter comunicare sia in scrittura che in lettura con la memoria.
- 13. o\_mem\_we: segnale di "WRITE ENABLE", necessario per poter scrivere nella memoria.

# 2 Architettura

## 2.1 Descrizione ad alto livello

Il modulo HW consiste di 6 differenti sottomoduli, ognuno realizzato per una specifica funzionalità necessaria al corretto comportamento del modulo stesso. Tali sottomoduli hanno tutti in comune il segnale di "CLOCK" proveniente dall'esterno del modulo HW e quello di "RESET", necessario per appunto resettare il contenuto/funzionamento dei singoli sottomoduli.



### 2.1.1 delayFF

Sottomodulo che si occupa di ritardare il segnale "W" di un ciclo di clock. Questo accorgimento si è reso necessario nella realizzazione del nostro progetto per come siamo andati a gestire la lettura dei singoli bit tramite i sottomoduli OutReg e OutAddr, e per come abbiamo gestito l'implementazione della FSM. Infatti, senza l'utilizzo di questo sottomodulo, abbiamo riscontrato il problema di non riuscire a leggere il primo bit della sequenza in ingresso tramite il segnale "W", cosa che andava ad influire sull'inserimento del dato letto da memoria nella giusta uscita, poichè la FSM necessita di leggere almeno un bit pari a 1 dal segnale start, per poter "abilitare" alla lettura il sottomodulo OutReg.

### 2.1.2 outReg

Sottomodulo molto semplice. Si occupa della lettura dei primi due bit della sequenza in ingresso sul segnale " $\mathbf{W}$ ". Tale combinazione di bit viene poi mandata in uscita al sottomodulo RegistryUpdater per poter scegliere il registro corretto nel quale andare a memorizzare il dato in uscita.

### 2.1.3 outAddr

Sottomodulo altrettanto semplice. Si occupa della lettura dei bit successi ai primi due presenti sul segnale "W", nel mentre che "START" si trova a 1. La lettura è stata implementata in modo che il modulo legga un bit alla volta dall'ingresso "W", inserendo sempre nella posizione 0 (quella più a destra) di un vettore, e ad ogni ciclo di "CLOCK", tale segnale temporaneo viene shiftato a sinistra, ripetendo l'azione fintanto che il sottomodulo rimane attivo (ovvero per come è stato costruita la FSM, fino a quando "START" ritorna a 0). Infine questo sottomodulo si occupa anche di inviare in uscita l'indirizzo di memoria letto per poter recuperare dalla memoria stessa il dato contenuto in tale indirizzo di memoria, che verrà spedito al Registry Updater.

### 2.1.4 RegistryUpdater

Sottomodulo che si occupa di gestire l'aggiornamento dei 4 registri visibili nell'immagine soprastante, utilizzati per memorizzare il valore che deve avere l'uscita fintanto che questa non venga sovrascritta, oppure ci sia un reset della macchina (in quel caso il contenuto dei registri viene completamente azzerato). L'aggiornamento viene gestito sulla base dell'indirizzo di due bit

letto dal sottomodulo OutReg, il quale determina quale dei 4 registri riceverà il dato letto nel frattempo dalla memoria per poter essere aggiornato. Oltre a tutto questo, il RegistryUpdater riceve in input il valore memorizzato all'interno dei 4 Registry, questo per fare in modo che tutti i registri, ad eccezione di quello da aggiornare, continuino a tenere memorizzato il dato che contenevano al ciclo di "CLOCK" precedente.

### 2.1.5 Registry8bit

Sottomodulo che si occupa di tenere memorizzato al suo interno il dato da mostrare in uscita, qualora "**DONE**" diventi pari a 1. Tale modulo riceve in ingresso il dato da memorizzare dal *RegistryUpdater* 

### 2.1.6 Controller

Sottomodulo che si occupa dell'implementazione della FSM nel componente, è l'effettivo controller del componente descritto nel progetto. La macchina spedificata è una macchina di Moore, dato che l'output è definito solo dallo stato in cui si trova la FSM, e non dall'input, basandosi sui segnali "START" e "RST" capisce in quali stati deve andare.

Lo stato S0 è lo stato iniziale, raggiunto dalla macchina dopo aver ricevuto il primo segnale di reset in input sul canale "RST. L'input della FSM che verrà considerato sarà allora solo il segnale "START", gli stati successivi allo stato S0 servono solo ad abilitare al momento corretto i diversi componenti, attraverso un vettore di segnali interni, l'uscita "outState".

I vari valori di "outState" possono essere mappati in una tabella, per capire a quale segnale corrisponde l'attivazione di quale componente.

- 000 -> Nessun componente interno abilitato
- 001 -> Componente "outReg" abilitato
- 010 -> Componente "outAddr" abilitato
- 011 -> Componente "registry Updater" abilitato
- 100 -> Segnale " $\mathbf{DONE}$ " messo pari ad 1, uscite passano da 0 al valore richiesto
- 101, 110, 111 -> Mai utilizzati, quindi DC



# 3 Risultati sperimentali

# 3.1 Report di sintesi

### 3.1.1 FSM Codificata

| State | New Encoding | Previous Encoding |
|-------|--------------|-------------------|
| s0    | 0000         | 0000              |
| sl    | 0001         | 0001              |
| s2    | 0010         | 0010              |
| s3    | 0011         | 0011              |
| s10   | 0100         | 1010              |
| s4    | 0101         | 0100              |
| s9    | 0110         | 1001              |
| s5    | 0111         | 0101              |
| s6    | 1000         | 0110              |
| s7    | 1001         | 0111              |
| s8    | 1010         | 1000              |

INFO: [Synth 8-3354] encoded FSM with state register 'current\_state\_reg' using encoding 'sequential'
in module 'controller'

Finished RTL Optimization Phase 2 : Time (s): cpu = 00:00:06 ; elapsed = 00:00:06 . Memory (MB): peak = 385.605 ; gain = 154.164

### 3.1.2 Statistiche RTL

## 3.1.3 Info Project Reti Logiche

```
Module project_reti_logiche
Detailed RTL Component Info :
+---Registers :
                   8 Bit Registers := 4
                  1 Bit Registers := 2
+---Muxes :
    2 Input
                   8 Bit
                               Muxes := 4
      2 Input
                   1 Bit
                               Muxes := 1
Module controller
Detailed RTL Component Info :
+---Muxes :
    19 Input 4 Bit
11 Input 3 Bit
                              Muxes := 1
                               Muxes := 1
```

### 3.1.4 Info Controller

## 3.1.5 Info delayFF

# 3.1.6 Info outReg

```
Module outReg
Detailed RTL Component Info :
+---Adders :
   2 Input
                32 Bit
                           Adders := 1
+---Registers :
                32 Bit
                        Registers := 1
                 2 Bit Registers := 2
+---Muxes :
              32 Bit
                            Muxes := 2
    2 Input
              2 Bit
1 Bit
      2 Input
                             Muxes := 6
      2 Input
                             Muxes := 5
```

### 3.1.7 Info outAddr

## 3.1.8 Info registryUpdater

## 3.1.9 Info registry8bit

# 3.2 Simulazioni

### 3.2.1 TB1

Uno dei primi test che abbiamo effettuato è stato verificare il comportamento della macchina tramite il testbench fornitoci in partenza, ed un test realizzato da noi:

### 3.2.2 TB2

Un caso particolare che abbiamo voluto testare, è il caso in cui vi sia la lettura di 18 uno consecutivi sul segnale "W", che comprende con il caso in cui venga letto un indirizzo di memoria di 16 bit pari a 1.



Figura 1: Caption

- 3.2.3 TB3
- 3.2.4 TB4
- 3.2.5 TB5
- 4 Conclusioni