# Proiect de Verificare în Proiectarea Circuitelor

Angelescu Denisa Andreea

# Cuprins

| 1. | Introducere                      | pg. 3  |
|----|----------------------------------|--------|
|    | -Tipul de Memorie Utilizat       | pg. 3  |
|    | -Scopul Proiectului              |        |
|    | -Testarea Memoriei cu UVM        | pg. 6  |
| 2. | Integrarea Memoriei în Testbench | pg. 7  |
|    | -Wrapper SystemVerilog           | pg. 7  |
|    | -Definirea Interfeței            | pg. 8  |
|    | -Testbench SystemVerilog         | pg. 9  |
| 3. | Componente Principale            | pg. 12 |
|    | -Testul                          | pg. 12 |
|    | -Sequence Items                  | pg. 17 |
|    | -Sequence                        | pg. 19 |
|    | -Driver                          | pg. 22 |
|    | -Agent                           | pg. 26 |
|    | -Environment                     | pg. 28 |
|    | -Monitor                         | pg. 30 |
| 4. | Forma de Undă                    | pg. 36 |
|    | -Scriere                         | pg. 36 |
|    | -Citire Asincronă                | pg. 38 |
|    | -Citire Sincronă                 | pg. 39 |
|    | -Reset Sincron                   | pg. 40 |
|    | -Reset Asincron                  | pg. 41 |
| 5. | Bibliografie                     | pa. 42 |

#### Introducere

#### <u>Tipul de Memorie Utilizat</u>

Acest proiect are ca obiectiv implementarea, simularea și verificarea unei memorii RAM cu un singur port, generată folosind Distributed Memory Generator de la AMD/Xilinx. Memoria este utilizată pentru stocarea și accesarea datelor, fiind testată cu Universal Verification Methodology (UVM) pentru a valida corectitudinea operațiilor de scriere, citire și reset.

#### Memorie RAM distribuită

- Implementată în FPGA folosind LUT-uri în loc de BRAM (Block RAM).
- Utilizată pentru aplicații cu latență mică și dimensiuni mici-medii de memorie.

#### Single Port RAM

- Are un singur port de acces, ceea ce înseamnă că citirea şi scrierea se fac pe aceleași semnale de adresă şi control.
- Nu permite acces simultan la două locații de memorie diferite (spre deosebire de Dual Port RAM).

# Scopul Proiectului

Prin intermediul acestui proiect, se urmăresc următoarele obiective principale:

- 1. Implementarea memoriei Single Port RAM
  - Memoria este configurată cu semnalele:
    - d Datele de scriere
    - a Adresa de acces
    - we Semnal de scriere
    - qspo ce Enable pentru citire sincronă
    - qspo rst Reset asincron pentru ieşirea sincronă
    - qspo srst Reset sincron pentru ieşirea sincronă

- clk Semnal de ceas pentru operațiile sincrone
- spo leşire asincronă
- qspo leşire sincronă



Fig. 1 PINOUT

- 2. Generarea corectă a memoriei cu Distributed Memory Generator
  - Alegerea configurației potrivite pentru porturi

| Input Options                           |                             |  |
|-----------------------------------------|-----------------------------|--|
| Non Registered    Registered            | Fig. 2 Opțiuni pentru Input |  |
| Output Options                          |                             |  |
| ○ Non Registered ○ Registered ● Both    |                             |  |
| Common Output CLK Single Port Output CE |                             |  |
| Common Output CE Dual Port Output CE    | Fig. 3 Optiuni pentru Ou    |  |

rig. 3 Opțiuni pentru Output

- spo (Simple Port Output asincronă)
- leşirea se actualizează imediat când adresa de intrare (a) se schimbă.
- Nu necesită semnal de ceas (clk).
- Utilă pentru acces rapid, dar poate introduce instabilitate dacă adresa se modifică frecvent.
  - qspo (Registered Synchronous Port Output sincronă)
- leşirea se actualizează doar la posedge clk, atunci când qspo ce este activ.

- Dacă qspo\_ce nu este activ, qspo păstrează ultima valoare citită.
- Utilizată pentru acces sincronizat, evitând erorile de competiție a datelor.
  - Definirea opțiunilor de reset și control

| Reset Options            |                                      | _                          |
|--------------------------|--------------------------------------|----------------------------|
| ✓ Reset QSPO             | Reset QDPO                           |                            |
| ✓ Synchronous Reset QSPC | Synchronous Reset QDPO               |                            |
| ce overrides             |                                      | _                          |
| CE Overrides Sync C      | ontrols   Sync Controls Overrides CE | Fig. 4 Opţiuni pentru Rese |

- 3. Verificarea corectitudinii funcționale folosind UVM
  - Testarea scrierii şi citirii:
    - Asigurarea că datele scrise la o adresă pot fi citite corect.
  - Verificarea resetului:
    - Resetul asincron (qspo\_rst) trebuie să șteargă ieșirea fără a depinde de ceas.
    - Resetul sincron (qspo\_srst) trebuie să reseteze ieşirea la primul posedge de clock.
  - o Analiza formelor de undă pentru a confirma comportamentul așteptat.
- 4. Depanarea și îmbunătățirea testbench-ului
  - Identificarea şi remedierea problemelor întâlnite în testare.
  - Compararea rezultatelor obţinute cu comportamentul aşteptat.

#### Testarea Memoriei cu UVM

Pentru a valida funcționalitatea memoriei, se utilizează Universal Verification Methodology (UVM), o metodologie standardizată pentru testarea modulelor hardware. Testarea include:

#### 1. Testarea scrierii și citirii

- Se scriu date într-o anumită locație de memorie.
- Se verifică dacă datele pot fi citite corect prin spo si gspo.

#### 2. Testarea resetului

- Reset asincron (qspo\_rst) → trebuie să reseteze qspo imediat, indiferent de clk.
- Reset sincron (qspo\_srst) → trebuie să reseteze qspo la primul posedge clk după activare.

#### 3. Analiza formelor de undă

- Se compară comportamentul observat în simulare cu cel așteptat.
- Se verifică dacă spo se modifică instantaneu şi qspo doar la posedge clk.

#### 4. Depanarea problemelor

- Se folosesc log-uri UVM (uvm\_info, uvm\_error) pentru a urmări execuția testului.
- Se analizează cazurile în care qspo sau spo nu se comportă conform specificaţiilor.

Prin această testare, se asigură că memoria funcționează corect în orice scenariu, fie că este vorba de scriere, citire sincronă sau resetare.

# Integrarea Memoriei în Testbench

#### Wrapper SystemVerilog pentru modulul dist mem gen 0 (VHDL)

Înainte de a începe verificarea memoriei, trebuie să ne asigurăm că aceasta poate fi utilizată corect într-un mediu de testare SystemVerilog-UVM. Deoarece Distributed Memory Generator generează automat un modul VHDL, dar testbench-ul nostru este scris în SystemVerilog, avem nevoie de un wrapper care să permită comunicarea dintre cele două limbaje.

Pentru a rezolva această problemă, am creat un wrapper SystemVerilog care înglobează modulul generat în VHDL (dist\_mem\_gen\_0) și îl expune în mod compatibil cu restul testbench-ului nostru.

Acest wrapper definește aceleași porturi de intrare și ieșire ca și memoria și doar instanțiază modulul VHDL, transmițând semnalele mai departe. Aceasta este o metodă utilizată frecvent pentru a permite interoperabilitatea între limbaje diferite în proiectele FPGA.

Mai exact, wrapper-ul primește următoarele semnale:

- Semnale de intrare: a (adresă), d (date), clk (ceas), we (scriere), qspo\_ce (enable pentru citire sincronă), qspo\_rst (reset asincron), qspo\_srst (reset sincron).
- Semnale de ieșire: spo (citire asincronă), qspo (citire sincronă).

După crearea acestui wrapper, putem folosi memoria exact ca un modul SystemVerilog nativ în testbench-ul nostru.

```
/ Wrapper SystemVerilog pentru modulul dist_mem_gen_0 (VHDL)
dule dist_mem_gen_wrapper (
  input logic [5:0] a,
        logic [15:0] d,
  input logic
  input logic
                      we,
  input logic
                      qspo_ce,
  input logic
                      qspo_rst,
  input logic
                      qspo_srst,
  output logic [15:0] spo,
  output logic [15:0] qspo
  // Instantierea modulului VHDL
  dist mem gen 0 inst dist mem gen 0 (
                 (a),
                 (d),
                 (clk),
                 (we),
      .qspo_rst (qspo_rst),
      .qspo_srst (qspo_srst),
                (spo),
                (qspo)
```

Fig. 5

## <u>Definirea Interfeței</u>

Pentru a simplifica conexiunile dintre testbench și DUT am definit o interfață (dist\_mem\_gen\_intf1). Aceasta include toate semnalele necesare pentru operarea memoriei, atât de intrare, cât și de ieșire. Ea interfață va fi instanțiată în testbench și utilizată pentru comunicarea dintre componente.

Utilizarea unei interfețe are mai multe avantaje, precum: claritate în cod (în loc să conectăm fiecare semnal manual, folosim interfața ca un pachet compact), ușurință în accesare (driver-ul, monitorul și alte componente UVM pot accesa toate semnalele prin această interfață), menținere și extindere facilă (dacă memoria se modifică, trebuie doar să schimbăm interfața, fără să modificăm întregul testbench).

```
interface dist_mem_gen_intf1();
   // Input
   logic [5:0] a;
   logic [15:0] d;
   logic
                 clk;
   logic
                 we;
   logic
                 qspo_ce;
   logic
                 qspo_rst;
                 qspo_srst;
   logic
   // Output
   logic [15:0] spo;
   logic [15:0] qspo;
endinterface : dist_mem_gen_intf1
```

Fig. 6

## Testbench SystemVerilog

Testbench-ul verifică funcționarea corectă a memoriei distribuite prin generarea de stimuli și analiza răspunsurilor acesteia. Acesta instanțiază memoria, creează semnalul de ceas și mapează semnalele printr-o interfață. Testele UVM sunt lansate prin run\_test(), trimițând tranzacții de citire și scriere, iar rezultatele sunt comparate cu valorile așteptate. Pentru a evita rularea infinită, testbench-ul impune o limită de cicluri de ceas și oprește simularea dacă testul nu se finalizează în timp util.

Testbench-ul începe prin definirea unității de timp și includerea bibliotecilor UVM necesare pentru simulare.

Definim semnalele de intrare și ieșire ale testbench-ului. Acestea vor fi utilizate pentru a comunica cu memoria instantiată.

```
module testbench();
logic [5:0] a;
logic [15:0] d;
logic
             clk;
logic
             we;
logic
             qspo_ce;
logic
             qspo_rst;
logic
             qspo_srst;
logic [15:0] spo;
logic [15:0] qspo;
                        Fig. 8
```

Instanțiem Distributive Memory Generator-ul printr-un wrapper, care permite utilizarea unui modul VHDL în SystemVerilog.

```
dist_mem_gen_wrapper dut_inst(.*);
Fig. 9
```

Instanțiem interfața de test, care va fi utilizată pentru a transmite semnale între componentele UVM și DUT.

```
dist_mem_gen_intf1 mem_intf();
Fig. 10
```

Generăm semnalul de ceas, care alternează la fiecare 10 ns, oferind un semnal de 50 MHz.

Conectăm semnalele testbench-ului la interfață, astfel încât acestea să fie utilizate corect de driver, monitor și DUT.

```
assign mem_intf.spo = spo;
assign mem_intf.qspo = qspo;
assign a = mem_intf.a;
assign d = mem_intf.d;
assign we = mem_intf.we;
assign qspo_ce = mem_intf.qspo_ce;
assign qspo_rst = mem_intf.qspo_rst;
assign qspo_srst = mem_intf.qspo_srst;
```

Fig. 12

Salvăm interfața în baza de date UVM, astfel încât toate componentele testbenchului să poată accesa semnalele.

```
initial begin
uvm_config_db#(virtual dist_mem_gen_intf1)::set(null, "", "dist_mem_gen_intf1", mem_intf); Fig. 13
```

Pornim testul UVM folosind run\_test(), care va executa secvențele de scriere, citire și resetare.

```
fork

begin

run_test("rw_test");

end

Fig. 14
```

Setăm o limită a ciclurilor de ceas, pentru a evita ca simularea să ruleze la infinit. Dacă testul nu s-a terminat după 100 de cicluri, simularea este oprită forțat.

Fig. 15

# Componente Principale



Fig. 16

#### Testul

Testul rw\_test este responsabil de verificarea funcționalității memoriei distribuite într-un mediu de simulare UVM. Acesta validează corectitudinea operațiunilor fundamentale ale memoriei, precum scrierea, citirea sincronă și asincronă, și mecanismele de resetare. Prin generarea și gestionarea tranzacțiilor de test, rw\_test se asigură că memoria reacționează corect la diferite scenarii și că datele sunt accesate corespunzător, conform specificațiilor.

Testul este construit folosind metodologia UVM și include o serie de sarcini (tasks) care trimit comenzi către memorie și colectează răspunsurile acesteia. Metodele de scriere (writeMemory), citire (readMemorySync, readMemoryAsync) și resetare (resetSync, resetAsync) sunt implementate pentru a acoperi toate modurile de operare ale memoriei. Prin run\_phase, testul coordonează execuția acestor operațiuni într-un mod secvențial, incluzând verificarea rezultatelor pentru a confirma integritatea datelor.

Obiectivul principal al testului rw\_test este de a detecta eventuale probleme în comportamentul memoriei și de a verifica respectarea specificațiilor de funcționare, oferind astfel o soluție robustă pentru validarea memoriei într-un sistem digital.

Testul rw test este definit ca o clasă care extinde uvm test. Acesta trebuie să fie înregistrat în UVM pentru a putea fi utilizat în testbench.

```
class rw_test extends uvm_test;
    `uvm_component_utils(rw_test)
   environment env;
                                Fig. 17
```

Liniile de mai sus declară clasa și înregistrează testul în UVM prin macro-ul uvm component utils(rw test). Variabila env reprezintă mediul de testare, care conține agentul UVM și componentele sale.

În build phase, testul creează instanța mediului de testare.

```
function new(input string name = "rw_test", uvm_component parent = null);
   super.new(name, parent);
endfunction : new
virtual function void build_phase(uvm_phase phase);
   env = environment::type_id::create("env", this);
endfunction : build_phase
```

l Fig. 18

Constructorul new apelează constructorul de bază pentru uvm\_test. În build phase, se creează o instanță a clasei environment, care conține agentul și componentele sale.

Metoda writeMemory creează o tranzacție de scriere, setează câmpurile necesare și o trimite către sequencer.

```
task writeMemory(int addr, int readdata);
   dist_mem_gen_rw_sequence writeSeq;
   writeSeq = dist_mem_gen_rw_sequence::type_id::create("writeSeq");
   writeSeq.we = 1:
   writeSeq.qspo_ce = 0;
   writeSeq.qspo_rst = 0;
   writeSeq.qspo srst = 0;
   writeSeq.a = addr;
   writeSeq.d = readdata;
    `uvm_info("WriteMemory Debug",
             $sformatf("Sending transaction: we=%0d, addr=%0h, data=%0h",
             writeSeq.we, writeSeq.a, writeSeq.d), UVM_LOW);
   writeSeq.start(env.agent.sequencer);
    `uvm_info("rw_test",
            $sformatf("Write issued: addr=%0h, data=%0h",
             addr, readdata), UVM_NONE);
 ndtask : writeMemory
```

Fig. 19

Aceasta creează un obiect writeSeq de tip dist\_mem\_gen\_rw\_sequence, setează semnalul we la 1 pentru a indica o scriere, atribuie adresa și datele, apoi trimite tranzacția către sequencer. Mesajele uvm info sunt utilizate pentru a urmări tranzacțiile.

În readMemorySync, datele sunt citite folosind semnalul qspo, care este valid doar la posedge clock.

```
task readMemorySync(input int addr, output int readdata);
    dist_mem_gen_rw_sequence readSeq;
    readSeq = dist_mem_gen_rw_sequence::type_id::create("readSeq");

readSeq.we = 0;
    readSeq.qspo_ce = 1;
    readSeq.qspo_rst = 0;
    readSeq.qspo_srst = 0;
    readSeq.a = addr;
    readSeq.d = env.agent.monitor.mem_intf.d;

readSeq.start(env.agent.sequencer);

readdata = env.agent.monitor.mem_intf.qspo;
endtask : readMemorySync
```

Fig. 20

Se setează we = 0 pentru a indica o citire şi qspo\_ce = 1 pentru a activa ieşirea sincronă qspo. După pornirea tranzacţiei, valoarea citită este preluată din qspo. Data este cea din trecut, îşi menţine valoarea.

În readMemoryAsync, ieșirea spo este utilizată pentru a returna imediat valoarea corespunzătoare adresei.

Fig. 21

În acest caz, qspo\_ce rămâne dezactivat (0), iar valoarea este returnată imediat prin spo, fără a fi necesar un posedge clock. Data este cea din trecut, își menține valoarea.

Această metodă activează qspo\_srst, resetând ieșirea qspo la 0.

```
task resetSync();
    dist_mem_gen_rw_sequence resetSeq;
    resetSeq = dist_mem_gen_rw_sequence::type_id::create("resetSyncSeq");

    resetSeq.we = 0;
    resetSeq.qspo_ce = 0;
    resetSeq.qspo_rst = 0;
    resetSeq.qspo_srst = 1;
    resetSeq.a = env.agent.monitor.mem_intf.a;

    resetSeq.start(env.agent.sequencer);
endtask : resetSync
```

Fig. 22

Aceasta trimite un semnal qspo\_srst = 1, iar resetarea se va realiza la următorul posedge clock. Adresa este cea din trecut, își menține valoarea.

Această metodă activează qspo\_rst, ceea ce duce la resetarea imediată a qspo.

```
task resetAsync();
    dist_mem_gen_rw_sequence resetSeq;
    resetSeq = dist_mem_gen_rw_sequence::type_id::create("resetAsyncSeq");

resetSeq.we = 0;
    resetSeq.qspo_ce = 0;
    resetSeq.qspo_rst = 1;
    resetSeq.qspo_srst = 0;
    resetSeq.a = env.agent.monitor.mem_intf.a;

resetSeq.start(env.agent.sequencer);
endtask : resetAsync
```

Fig. 23

Resetarea are loc imediat ce qspo\_rst este activat. Adresa este cea din trecut, își menține valoarea.

În metodele writeMemory și readMemory, variabila readdata are un rol diferit în funcție de operația efectuată.

- În writeMemory, readdata conține valoarea care trebuie scrisă în memorie.
   Aceasta este atribuită câmpului d al tranzacției writeSeq, reprezentând datele care vor fi stocate la adresa addr.
- În readMemorySync şi readMemoryAsync, readdata este folosit pentru a colecta valoarea citită din memorie. În cazul citirii sincrone (qspo), aceasta este actualizată la posedge clock, iar în cazul citirii asincrone (spo), aceasta este disponibilă imediat.

În run\_phase, sunt executate secvențele de scriere, citire și resetare într-un mod organizat. Testul verifică scrierea, citirea și efectul resetării asupra memoriei:

```
writeMemory(6'h10, 16'h1234);
#25
writeMemory(6'h20, 16'h5678); Fig. 24
```

Aceste două instrucțiuni scriu valorile 0x1234 și 0x5678 la adresele 0x10 și 0x20. După fiecare scriere, testul așteaptă un anumit timp înainte de următoarea operație.

```
readMemoryAsync(6'h10, readdata);
$display("Async Read Data after Write 1: %0h", readdata);
Fig. 25
```

Această secvență citește valoarea din adresa 0x10 în mod asincron și o afișează. Deoarece citirea asincronă returnează datele imediat, readdata este actualizat instant cu valoarea citită.

```
resetSync();
#40
readMemorySync(6'h10, readdata);
$display("Sync Read Data: %0h", readdata);
Fig. 26
```

Această secvență activează resetul sincron, așteaptă 40ns, apoi încearcă să citească din adresa 0x10. După resetare, valoarea citită qspo ar trebui să fie 0 sincron cu ceasul, indicând că resetul a fost aplicat corect.

```
resetAsync();
#13
readMemorySync(6'h30, readdata);
$display("Sync Read Data: %0h", readdata);
Fig. 27
```

Aici, resetul asincron este aplicat, ceea ce ar trebui să reseteze imediat valoarea memoriei. După 13ns, testul efectuează o citire sincronă pentru a verifica dacă resetul a avut efect.

Variabila readdata este utilizată pentru a colecta și verifica datele din memorie în timpul citirii. În cadrul scrierii, aceasta conține datele care trebuie stocate. Prin combinația dintre scriere, citire și resetare, testul rw\_test validează corectitudinea memoriei și comportamentul acesteia în diferite scenarii.

#### Sequence Items

Clasa dist\_mem\_gen\_seq\_items1 reprezintă elementul de bază al tranzacțiilor utilizate în testarea memoriei distribuite. Aceasta definește structura unei tranzacții, incluzând adresa, datele și semnalele de control necesare pentru scriere, citire și resetare.

Fiind o extensie a uvm\_sequence\_item, această clasă permite generarea de stimuli într-un mod abstract, facilitând interacțiunea dintre secvențele de test (sequence) și driver. Astfel, sequencer-ul poate trimite aceste obiecte de tranzacție către driver, care le va converti în semnale hardware reale aplicate memoriei.

În plus, clasa include funcționalități de inițializare și debugging, permițând verificarea valorilor tranzacțiilor printr-o metodă de conversie în string. Acest aspect este esențial pentru monitorizarea și analiza comportamentului memoriei în timpul simulării.

Scopul dist\_mem\_gen\_seq\_items1 este de a asigura un model clar şi organizat pentru schimbul de date între componentele testbench-ului, contribuind astfel la verificarea funcționării corecte a memoriei distribuite.

Această clasă extinde uvm\_sequence\_item, ceea ce înseamnă că este utilizată pentru a transporta date între sequencer și driver. Tranzacțiile definite în această clasă sunt esențiale pentru a modela operațiile memoriei.

class dist\_mem\_gen\_seq\_items1 extends uvm\_sequence\_item;
Fig. 28

Această macro înregistrează clasa în cadrul UVM, permiţând utilizarea sa în secvenţe (uvm sequence).

```
`uvm_object_utils(dist_mem_gen_seq_items1) Fig. 29
```

Aceasta este zona în care sunt declarate variabilele pentru tranzacții:

- **spo și qspo** Acestea reprezintă ieșirile memoriei, spo fiind citirea asincronă, iar qspo citirea sincronă.
- d Datele care trebuie scrise în memorie. Este declarată ca rand, ceea ce înseamnă că poate fi randomizată pentru testare.
- we Semnalul de scriere (Write Enable).
- qspo\_ce Semnal pentru activarea citirii sincronizate.
- qspo\_rst şi qspo\_srst Semnale de reset, asincron şi sincron.
- a Adresa memoriei unde se citeşte sau scrie.

```
logic [15:0] spo, qspo;
rand logic [15:0] d;
rand logic we, qspo_ce, qspo_rst, qspo_srst;
rand logic [5:0] a;
Fig. 30
```

Constructorul inițializează variabilele tranzacției cu valori implicite (0). Astfel, atunci când se creează un nou obiect de tranzacție, acesta nu conține valori nedeterminate.

```
function new(input string name="dist_mem_gen_seq_items1");
    super.new(name);

spo = 0;
    qspo = 0;
    d = 0;
    we = 0;
    a = 0;
    qspo_ce = 0;
    qspo_rst = 0;
    qspo_srst = 0;
endfunction : new
Fig. 31
```

Această funcție returnează o reprezentare sub formă de text a tranzacției, utilă pentru debug în timpul simulării.

```
function string convert2string();
    string outputstring = "";

outputstring = $sformatf("%s\n\t * spo=%0h", outputstring, spo);
    outputstring = $sformatf("%s\n\t * qspo=%0h", outputstring, qspo);
    outputstring = $sformatf("%s\n\t * a=%0h", outputstring, a);
    outputstring = $sformatf("%s\n\t * we=%0h", outputstring, we);
    outputstring = $sformatf("%s\n\t * d=%0h", outputstring, d);
    outputstring = $sformatf("%s\n\t * qspo_ce=%0h", outputstring, qspo_ce);
    outputstring = $sformatf("%s\n\t * qspo_rst=%0h", outputstring, qspo_rst);
    outputstring = $sformatf("%s\n\t * qspo_srst=%0h", outputstring, qspo_srst);
    return outputstring;
endfunction
```

#### <u>Sequence</u>

Clasa dist\_mem\_gen\_rw\_sequence este o secvență UVM care generează tranzacții pentru interacțiunea cu memoria distribuită. Aceasta este responsabilă de inițializarea și trimiterea operațiunilor către sequencer, care le va direcționa către driver pentru a fi aplicate efectiv asupra memoriei. Secvența permite testarea scrierii, citirii sincrone și resetării memoriei, asigurând o verificare completă a funcționalității acesteia.

Această secvență este utilizată în cadrul testului UVM pentru a crea tranzacțiile necesare și a le livra agentului de testare. Prin definirea variabilelor care controlează operațiile memoriei, dist\_mem\_gen\_rw\_sequence facilitează execuția testelor și permite verificarea răspunsului memoriei în diferite condiții de operare. Obiectivul principal este de a modela stimuli reali care să permită validarea corectă a funcționalității memoriei distribuite.

Macro-ul uvm\_object\_utils(dist\_mem\_gen\_rw\_sequence) înregistrează clasa în UVM, permiţând utilizarea sa în testbench şi facilitând crearea de instanţe.

```
class dist_mem_gen_rw_sequence extends uvm_sequence #(dist_mem_gen_seq_items1); Fig. 33
```

Macro-ul uvm\_object\_utils(dist\_mem\_gen\_rw\_sequence) înregistrează clasa în UVM, permiţând utilizarea sa în testbench şi facilitând crearea de instanţe.

```
`uvm_object_utils(dist_mem_gen_rw_sequence) Fig. 34
```

Variabilele we, qspo\_ce, qspo\_rst, qspo\_srst, a şi d sunt utilizate pentru configurarea tranzacţiilor generate. we controlează scrierea în memorie, qspo\_ce activează citirea sincronizată, qspo\_rst şi qspo\_srst sunt semnale de resetare asincronă şi sincronă, iar a şi d reprezintă adresa şi datele utilizate în operaţiile asupra memoriei.

```
bit we;
bit qspo_ce;
bit qspo_rst;
bit qspo_srst;
int a;
int d; Fig. 35
```

Constructorul new apelează constructorul clasei de bază și permite crearea de instanțe ale secvenței.

```
function new(input string name = "dist_mem_gen_rw_sequence");
    super.new(name);
endfunction : new
Fig. 36
```

Metoda body este responsabilă pentru generarea și trimiterea tranzacțiilor. În această metodă, se creează un obiect rw\_item de tip dist\_mem\_gen\_seq\_items1, care este inițializat cu valorile setate în testul rw\_test.

```
virtual task body();
    dist_mem_gen_seq_items1 rw_item;
    rw_item = dist_mem_gen_seq_items1::type_id::create("rw_item");
    Fig. 37
```

Tranzacția este transmisă către sequencer prin start\_item, după care sunt atribuite valorile corespunzătoare pentru adresă, date și semnalele de control.

```
start_item(rw_item);

rw_item.a = a;

rw_item.d = d;

rw_item.we = we;

rw_item.qspo_ce = qspo_ce;

rw_item.qspo_rst = qspo_rst;

rw_item.qspo_srst = qspo_srst;
Fig. 38
```

Un mesaj uvm\_info afișează detalii despre tranzacție, inclusiv adresa, datele și semnalul we, permițând verificarea corectitudinii execuției.

```
`uvm_info("dist_mem_gen_rw_sequence",

$sformatf("Before finish_item: addr=%0h, data=%0h, we=%0h",
a, d, we), UVM_NONE);

Fig. 39
```

Tranzacția este finalizată prin finish\_item, ceea ce semnalează sequencer-ului că aceasta poate fi preluată și trimisă către driver.

```
finish_item(rw_item);
endtask : body
Fig. 40
```

#### Driver

Clasa dist\_mem\_gen\_driver este responsabilă pentru preluarea tranzacţiilor din sequencer şi aplicarea acestora asupra interfeţei memoriei distribuite. Aceasta transformă tranzacţiile abstracte generate de secvenţă în stimuli reali care sunt trimişi către DUT (Device Under Test).

Driverul funcționează într-un mod continuu, așteptând tranzacții din secvență și actualizând semnalele interfeței corespunzător pentru fiecare operație. În cadrul metodei run\_phase, acesta identifică tipul de tranzacție primită și acționează în consecință:

• Dacă tranzacția este o **scriere**, driverul setează adresa și datele în memorie, activează semnalul we, apoi îl dezactivează după un ciclu de ceas.

- Dacă tranzacția este o citire sincronizată, driverul setează adresa și activează qspo\_ce, așteptând un ciclu de ceas pentru a obține valoarea citită.
- Dacă tranzacția este o citire asincronă, driverul setează adresa şi citeşte imediat valoarea spo.
- Dacă tranzacția este un reset sincron, driverul activează qspo\_srst, așteaptă un ciclu de ceas, apoi îl dezactivează.
- Dacă tranzacția este un reset asincron, driverul activează qspo\_rst şi îl dezactivează imediat după un delay minim.

La fiecare tranzacție procesată, driverul semnalează secvenței că poate trimite următoarea tranzacție, asigurând un flux continuu al execuției. În plus, acesta folosește mesaje uvm\_info pentru a afișa informații despre fiecare tranzacție, facilitând procesul de debug și verificare.

Această clasă extinde uvm\_driver și specifică tipul de tranzacție procesată, care este dist\_mem\_gen\_seq\_items1. Driverul este componenta UVM care preia tranzacțiile din secvență și le convertește în stimuli reali pentru DUT.

```
class dist_mem_gen_driver extends uvm_driver#(dist_mem_gen_seq_items1);
Fig. 41
```

Această macro înregistrează driverul în UVM, permițând utilizarea sa în testbench și facilitând crearea de instanțe.

```
`uvm_component_utils(dist_mem_gen_driver)
```

Constructorul clasei este necesar pentru inițializarea driverului și apelarea constructorului clasei părinte.

```
function new(input string name = "", uvm_component parent = null);
    super.new(name,parent);
endfunction : new
Fig. 43
```

Metoda run\_phase este responsabilă pentru rularea driverului în timpul fazei run din UVM. Aici, driverul așteaptă tranzacții, le procesează și le aplică pe interfață. Variabila dist mem gen item reprezintă o tranzacție primită de la sequencer.

```
virtual task run_phase(uvm_phase phase);
    dist_mem_gen_seq_items1 dist_mem_gen_item;
    Fig. 44
```

Se creează o referință la interfața de test pentru a permite driverului să controleze semnalele memoriei. Interfața este extrasă din uvm\_config\_db, ceea ce permite accesul la ea din alte componente.

```
virtual dist_mem_gen_intf1 mem_intf;
uvm_config_db#(virtual dist_mem_gen_intf1)::get(null, "", "dist_mem_gen_intf1", mem_intf);
Fig. 45
```

Driverul rulează într-un loop continuu, așteptând și preluând tranzacții din sequencer. Odată ce o tranzacție este primită, aceasta este stocată în dist\_mem\_gen\_item.

```
forever begin

seq_item_port.get_next_item(dist_mem_gen_item);
Fig. 46
```

Un mesaj de debug este afișat pentru a indica detaliile tranzacției primite.

```
`uvm_info("dist_mem_gen_driver",

$sformatf("Received new item: %s",

dist_mem_gen_item.convert2string()),

UVM_NONE)

Fig. 47
```

Semnalele interfeței sunt actualizate conform valorilor din tranzacția primită, în special până la primul posedge clk care oricum le inițializează. Acest pas pregătește stimulii pentru a fi aplicați asupra DUT.

```
mem_intf.we = dist_mem_gen_item.we;
mem_intf.qspo_ce = dist_mem_gen_item.qspo_ce;
mem_intf.qspo_rst = dist_mem_gen_item.qspo_rst;
mem_intf.qspo_srst = dist_mem_gen_item.qspo_srst;
mem_intf.a = dist_mem_gen_item.a;
mem_intf.d = dist_mem_gen_item.d;
Fig. 48
```

Dacă tranzacția primită este o scriere (we == 1), driverul așteaptă un ciclu de clock, setează adresa și datele pe interfață, activează we și apoi îl dezactivează după un ciclu de clock.

```
if (dist_mem_gen_item.we == 1) begin
    @(posedge mem_intf.clk);
    `uvm_info("dist_mem_gen_driver", "CLK edge detected in driver.", UVM_NONE);

mem_intf.d = dist_mem_gen_item.d;
mem_intf.a = dist_mem_gen_item.a;
mem_intf.we = 1;

mem_intf.qspo_ce = 0;
mem_intf.qspo_rst = 0;
mem_intf.qspo_srst = 0;

@(posedge mem_intf.clk);
mem_intf.we = 0;

`uvm_info("dist_mem_gen_driver", "Write operation completed successfully.", UVM_NONE);
end
```

Fig. 49

Dacă tranzacția primită este un reset sincron (qspo\_srst == 1), driverul activează semnalul qspo\_srst, așteaptă un ciclu de clock și apoi îl dezactivează.

```
else if (dist_mem_gen_item.qspo_srst == 1) begin
  @(posedge mem_intf.clk);
  mem_intf.qspo_srst = 1;
  @(posedge mem_intf.clk);
  mem_intf.qspo_srst = 0;
  `uvm_info("dist_mem_gen_driver", "Synchronous reset applied to QSPO.", UVM_NONE);
end
Fig. 50
```

Dacă tranzacția este un reset asincron (qspo\_rst == 1), driverul activează qspo\_rst, așteaptă un delay minim și apoi îl dezactivează imediat.

```
else if (dist_mem_gen_item.qspo_rst == 1) begin
    mem_intf.qspo_rst = 1;
    #1;
    mem_intf.qspo_rst = 0;

`uvm_info("dist_mem_gen_driver", "Asynchronous reset applied to QSPO.", UVM_NONE);
end
```

Fig. 51

Dacă tranzacția primită este o citire sincronizată (qspo\_ce == 1), driverul setează adresa, activează qspo\_ce, așteaptă un ciclu de clock și stochează valoarea citită în qspo.

Dacă tranzacția este o citire asincronă, driverul setează adresa și citește imediat valoarea spo de la ieșirea memoriei.

După procesarea tranzacției, driverul semnalează sequencer-ului că a finalizat tranzacția, astfel încât următoarea să poată fi trimisă.

```
seq_item_port.item_done();Fig. 54
```

#### **Agent**

Clasa dist\_mem\_gen\_agent reprezintă un agent UVM care grupează componentele esențiale pentru interacțiunea cu DUT: driver, sequencer și monitor. Agentul facilitează comunicarea între aceste componente și gestionează fluxul de tranzacții dintre secvență si dispozitivul testat.

În cadrul metodei build\_phase, sunt create instanțele pentru sequencer, driver și monitor, asigurând astfel structura completă a agentului. sequencer este responsabil pentru generarea tranzacțiilor și trimiterea acestora către driver, care le aplică asupra DUT. monitor observă semnalele DUT și colectează datele pentru analiză.

Metoda connect\_phase stabilește conexiunile dintre driver și sequencer, permițând trimiterea tranzacțiilor către DUT. Aceasta se asigură că seq\_item\_port al driver este conectat la seq\_item\_export al sequencer, permițând preluarea și execuția tranzacțiilor generate. Agentul joacă un rol crucial în organizarea testbench-ului UVM, oferind o interfață clară între generarea de stimuli și verificarea comportamentului DUT.

Clasa moștenește uvm\_agent, ceea ce înseamnă că acționează ca un container pentru componentele de bază ale unui testbench UVM.

```
class dist_mem_gen_agent extends uvm_agent; Fig. 55
```

Această macro înregistrează clasa în UVM, permiţând crearea de instanţe şi utilizarea sa în testbench.

```
`uvm_component_utils(dist_mem_gen_agent) Fig. 56
```

Aici sunt declarate componentele agentului. monitor este responsabil pentru captarea și analizarea semnalelor DUT, driver aplică tranzacțiile asupra DUT, iar sequencer generează și trimite tranzacții către driver.

```
dist_mem_gen_monitor monitor;
dist_mem_gen_driver driver;
uvm_sequencer #(dist_mem_gen_seq_items1) sequencer;
Fig. 57
```

Constructorul new apelează constructorul clasei părinte și permite crearea de instanțe ale agentului.

```
function new(input string name = "", uvm_component parent = null);
    super.new(name, parent);
endfunction : new
Fig. 58
```

Metoda build\_phase iniţializează componentele agentului. Se creează instanţe pentru sequencer, driver şi monitor, asigurându-se că acestea sunt disponibile pentru test.

```
virtual function void build_phase(uvm_phase phase);
    sequencer = uvm_sequencer#(dist_mem_gen_seq_items1)::type_id::create("sequencer", this);
    driver = dist_mem_gen_driver::type_id::create("driver", this);
    monitor = dist_mem_gen_monitor::type_id::create("monitor", this);
endfunction
Fig. 59
```

Metoda connect\_phase este utilizată pentru a conecta componentele agentului. Se afișează un mesaj de debug care indică faptul că conexiunile au fost stabilite.

Această linie de cod conectează seq\_item\_port al driver la seq\_item\_export al sequencer. Astfel, tranzacțiile generate de sequencer sunt transmise către driver, care le aplică pe DUT.

```
driver.seq_item_port.connect(sequencer.seq_item_export); Fig. 61
```

#### **Environment**

Clasa environment reprezintă componenta UVM care grupează toate elementele esențiale pentru testare, inclusiv agent și scoreboard. Aceasta asigură organizarea ierarhică a testbench-ului și gestionează fluxul de date între componentele sale.

Agentul (dist\_mem\_gen\_agent) este responsabil pentru generarea, aplicarea și monitorizarea tranzacțiilor asupra DUT, iar scoreboard se ocupă cu verificarea corectitudinii rezultatelor.

În metoda build\_phase, sunt create instanțele agentului și scoreboard-ului, asigurând structura testbench-ului. Metoda connect\_phase realizează conexiunea între monitor-ul agentului și scoreboard, permițând verificarea datelor colectate în timpul testării.

Clasa environment servește drept nivel de integrare pentru componentele de testare, facilitând verificarea completă a DUT prin gestionarea fluxului de date între generatorul de stimuli, monitor și verificator.

Clasa environment extinde uvm\_env, ceea ce înseamnă că acționează ca un container pentru componentele de testare, cum ar fi agentul și scoreboard.

Această macro permite înregistrarea clasei în UVM, facilitând crearea de instanțe și utilizarea sa în testbench.

Aici sunt declarate componentele mediului de testare. agent este responsabil pentru generarea, aplicarea și monitorizarea tranzacțiilor asupra DUT, iar sb (scoreboard) verifică dacă datele obținute corespund rezultatelor așteptate.

Aceasta este funcția constructor, care apelează constructorul clasei de bază (uvm\_env) și permite inițializarea instanțelor de environment.

În build\_phase, sunt create instanțele agentului și scoreboard-ului. build\_phase este utilizată pentru inițializarea componentelor UVM înainte de rularea testului. Se afișează un mesaj de debug pentru a confirma apelarea acestei metode.

```
virtual function void build_phase(uvm_phase phase);
    `uvm_info("DEBUG", "The build_phase of the environment was called", UVM_NONE)

agent = dist_mem_gen_agent::type_id::create("agent", this);
    sb = scoreboard::type_id::create("sb", this);
endfunction
Fig. 66
```

Metoda connect\_phase realizează conexiunea dintre monitor-ul agentului și scoreboard, permițând transferul datelor capturate în monitor către scoreboard pentru verificare. Aceasta asigură că informațiile colectate sunt analizate corect și comparate cu rezultatele așteptate.

```
virtual function void connect_phase(uvm_phase phase);
    agent.monitor.monitor_analysis_port.connect(sb.monitor_analysis_port);
endfunction : connect_phase
Fig. 67
```

#### Monitor

Clasa dist\_mem\_gen\_monitor este responsabilă pentru capturarea și analizarea tranzacțiilor care se desfășoară în timpul testării distribuției memoriei. Aceasta observă semnalele de intrare și ieșire ale DUT și le trimite către scoreboard pentru verificare.

Această clasă include un covergroup, care ajută la măsurarea acoperirii testării prin colectarea informațiilor despre operațiile efectuate asupra memoriei. monitor\_analysis\_port este utilizat pentru a transmite tranzacțiile capturate către scoreboard, asigurând astfel validarea corectitudinii execuției.

În build\_phase, sunt iniţializate instanţele necesare pentru monitorizare şi acoperire. În connect\_phase, interfaţa virtuală este asociată pentru a permite accesul la semnalele DUT. run\_phase declanşează monitorizarea operaţiilor, pornind procesele paralele pentru scriere, citire sincronă, citire asincronă şi resetare.

Metodele monitor\_write, monitor\_sync\_read şi monitor\_async\_read sunt responsabile pentru detectarea operaţiunilor de scriere şi citire, capturând datele şi trimiţându-le către scoreboard. monitor\_resets se ocupă de observarea reseturilor asincrone şi sincronizate.

La final, report\_phase afișează rezultatele acoperirii testului, oferind o măsură a eficienței testării și a tranzacțiilor capturate în timpul execuției.

Acest covergroup colectează informații despre tranzacțiile testate, cum ar fi adresele accesate, datele scrise, activarea semnalelor de scriere și citire. Într-o etapă ulterioară, aceste date vor fi utilizate pentru a îmbunătăți testarea și pentru randomizarea stimulilor astfel încât toate scenariile relevante să fie acoperite.

Fig. 68

Aceasta definește monitorul și include un port de analiză (monitor\_analysis\_port) pentru a trimite tranzacțiile observate către scoreboard. De asemenea, instanțiază covergroup pentru a colecta date despre testare.

```
class dist_mem_gen_monitor extends uvm_monitor;
    `uvm_component_utils(dist_mem_gen_monitor)
    uvm_analysis_port #(dist_mem_gen_seq_items1) monitor_analysis_port;
    cg_transactions my_cg;
    virtual dist_mem_gen_intf1 mem_intf;
Fig. 69
```

Constructorul clasei inițializează monitorul și permite integrarea sa în ierarhia UVM.

```
function new(input string name = "", uvm_component parent = null);
    super.new(name, parent);
endfunction : new
Fig. 70
```

În această etapă, este creat portul de analiză și se inițializează instanța covergroup, care va colecta datele în timpul rulării testului.

Această funcție asociază monitorul cu interfața virtuală a DUT-ului, permițând astfel accesul la semnalele acestuia.

```
virtual function void connect_phase(uvm_phase phase);
   super.connect_phase(phase);
   uvm_config_db#(virtual dist_mem_gen_intf1)::get(this, "",
        "dist_mem_gen_intf1", mem_intf);
endfunction
Fig. 72
```

Monitorul rulează în paralel mai multe procese pentru capturarea diferitelor tipuri de tranzacții: scriere, resetare, citire sincronă și citire asincronă.

```
virtual task run_phase(uvm_phase phase);
  fork
       monitor_write();
      monitor_resets();
      monitor_sync_read();
      monitor_async_read();
      join
endtask : run_phase
Fig. 73
```

Această metodă detectează scrierile în memorie (sincronizate cu posedge clk) și le transmite către scoreboard. În același timp, înregistrează datele în covergroup, care va fi folosit ulterior pentru îmbunătățirea testării.

```
task monitor_write();
    forever begin
       dist_mem_gen_seq_items1 observed_item;
       repeat(1)@(posedge mem_intf.clk);
        if (mem_intf.we == 1) begin
            observed_item = dist_mem_gen_seq_items1::type_id::create(
                "observed_item_write");
            observed_item.we = mem_intf.we;
            observed_item.d = mem_intf.d;
            observed_item.a = mem_intf.a;
            `uvm_info("dist_mem_gen_monitor",
                $sformatf("Write operation observed: Address=%0h, Data=%0h",
                observed_item.a, observed_item.d), UVM_NONE);
            my_cg.sample();
            monitor_analysis_port.write(observed_item);
        end
    end
endtask : monitor_write
```

Fig. 74

Aceasta capturează operațiile de citire sincronă (când este activă citirea sincronă -> qspo\_ce = 1), trimite tranzacțiile observate la scoreboard și le înregistrează în covergroup.

```
task monitor_sync_read();
    forever begin
       dist_mem_gen_seq_items1 observed_item;
       @(posedge mem_intf.clk);
        if (mem_intf.qspo_ce) begin
            observed_item = dist_mem_gen_seq_items1::type_id::create(
                "observed_item_sync_read");
            observed_item.qspo_ce = mem_intf.qspo_ce;
            observed_item.a = mem_intf.a;
            observed_item.qspo = mem_intf.qspo;
            `uvm info("dist mem gen monitor",
                $sformatf("Synchronous read operation observed: Address=%0h, Data=%0h",
                observed_item.a, observed_item.qspo), UVM_NONE);
            my_cg.sample();
            monitor_analysis_port.write(observed_item);
        end
    end
endtask : monitor_sync_read
```

Fig. 75

Monitorizează citirile asincrone (nu există nicio constrângere aici, decât că se modifică la schimbarea adresei sau a ieșirii din DUT) și înregistrează tranzacțiile observate pentru a verifica dacă testele acoperă toate scenariile necesare.

Fig. 76

Monitorizează semnalele de reset (se așteaptă un posedge pentru qspo\_rst si qspo\_srst; dacă qspo\_rst = 1, atunci resetul este activat imediat; dacă qspo\_srst = 1, atunci se așteaptă un posedge clk pentru ca resetul sa fie activat) și asigură că acestea sunt detectate corect în cadrul testului.

```
task monitor_resets();
   dist_mem_gen_seq_items1 observed_item;
   forever begin
       @(posedge mem_intf.qspo_rst or posedge mem_intf.qspo_srst);
       observed_item = dist_mem_gen_seq_items1::type_id::create(
           "observed_item_reset");
       if (mem_intf.qspo_rst == 1) begin
            `uvm_info("dist_mem_gen_monitor", "Asynchronous reset observed.", UVM_NONE);
       end
       if (mem_intf.qspo_srst == 1) begin
           @(posedge mem_intf.clk);
            `uvm_info("dist_mem_gen_monitor", "Synchronous reset observed.", UVM_NONE);
       end
       my_cg.sample();
       monitor_analysis_port.write(observed_item);
endtask : monitor_resets
```

Fig. 77

La finalul testului, se raportează procentajul de acoperire colectat de covergroup, ceea ce ajută la analiza calității testării și identificarea cazurilor neacoperite.

```
virtual function void report_phase(uvm_phase phase);
   super.report_phase(phase);
   $display("Coverage Results in Monitor:");
   `uvm_info("COVERAGE_INFO",
        $sformatf("Coverage on my_cg=%d", my_cg.get_inst_coverage()),
        UVM_NONE);
endfunction : report_phase
```

Fig. 78

#### Forma de Undă

#### **Scriere**

```
writeMemory(6'h10, 16'h1234);
#25
writeMemory(6'h20, 16'h5678);
#55
writeMemory(6'h30, 16'h9ABC);
repeat(4) @(posedge env.agent.monitor.mem_intf.clk);
writeMemory(6'h2, 16'h3);
#15
writeMemory(6'h20, 16'h3333);
#15
writeMemory(6'h30, 16'h0101);
#15
Fig. 79
```



Fig. 80

Până la primul posedge al semnalului de ceas, valorile semnalelor sunt definite conform Fig. 48. La momentul 10ns, semnalul we devine 1, ceea ce declanșează o operație de scriere la adresa 10h. Înainte de acest moment, ieșirea spo avea valoarea 0, iar imediat după scriere se actualizează la 1234h. we va deveni 0 la următorul posedge clk, așa cum se poate vedea în Fig. 49. La 55ns, we revine la 0 și apoi urcă din nou la 1, ceea ce indică o nouă scriere, de această dată la adresa 20h. Când adresa de acces se schimbă de la 10h la 20h, spo trece temporar la 0, deoarece adresa 20h nu fusese utilizată anterior. Scrierile efective în memorie au loc doar la posedge-ul semnalului de ceas, moment în care we este activ. Astfel, la 70ns, scrierea la adresa 20h este finalizată, iar spo este actualizat la 5678h.



Fig. 81

La 250ns, în momentul în care apare un posedge al semnalului de ceas, we este încă 0, ceea ce înseamnă că nicio operație de scriere nu este inițiată în acel ciclu de ceas. După acest moment, we devine 1, indicând intenția de a efectua o scriere. Adică we se află în tranziția de la 0 la 1. Totuși, deoarece scrierile în memorie sunt procesate doar la posedge-ul ceasului, această operație va fi luată în considerare abia la următorul posedge al clk. Astfel, la 270ns, scrierea este efectuată, iar valoarea corespunzătoare este înregistrată în memorie.



Fig. 82

#### Citire Asincronă

```
// Citire asincronă
readMemoryAsync(6'h10, readdata);
$display("Async Read Data after Write 1: %0h", readdata);
#35;
readMemoryAsync(6'h20, readdata);
$display("Async Read Data after Write 2: %0h", readdata);
#20;
readMemoryAsync(6'h30, readdata);
$display("Async Read Data after Write 3: %0h", readdata);
#30;
readMemoryAsync(6'h10, readdata);
$display("Async Read Data after Write 1: %0h", readdata);
#45;
```

Fig. 83



Fig. 84

La 385ns, adresa se schimbă la 10h și se inițiază o citire asincronă prin spo. Se poate observa că această operație are loc instantaneu, fără a fi necesară așteptarea unui ciclu de ceas. Același comportament se observă și pentru următoarele două citiri asincrone. Ultima citire asincronă, realizată la 470ns, confirmă că posedge clk nu influențează procesul de citire și că acesta se desfășoară corect. De asemenea, se remarcă faptul că valoarea datelor rămâne constantă, menținând ultima valoare validă.

## Citire sincronă

```
// Citire sincronă
readMemorySync(6'h2, readdata);
$display("Sync Read Data: %0h", readdata);
#60
readMemorySync(6'h20, readdata);
$display("Sync Read Data: %0h", readdata);
#55
```

Fig. 85



Fig. 86

La 515ns, qspo\_ce din 0 devine 1 și la următorul posedge clk se va lua in considerare citirea sincronă qspo, adică la 530ns, iar această ieșire va deveni la 530ns valoarea de la adresa corespunzătoare, până la următoarea comandă. qspo\_ce devine 0 la următorul posedge clk, așa cum se poate observa în Fig. 52. La 610ns pe posedge clk, qspo\_ce se află în tranziția de la 0 la 1, deci qspo\_ce = 1 la următorul posedge clk și atunci o să se și vadă și update ul lui qspo.

#### **Reset Sincron**

```
// Reset sincron
resetSync();
#40
readMemorySync(6'h10, readdata);
$display("Sync Read Data: %0h", readdata);
#20
resetSync();
#10
readMemorySync(6'h20, readdata);
$display("Sync Read Data: %0h", readdata);
#7
```

Fig. 87



Fig. 88

La momentul 705ns, semnalul qspo\_srst este activat, însă acesta nu afectează imediat ieșirea qspo, deoarece resetul sincron necesită un ciclu de ceas pentru a fi luat în considerare. Astfel, la 710ns, pe următorul posedge clk, qspo este setat la 0 și rămâne în această stare până la o actualizare ulterioară. qspo\_srst devine 0 la următorul posedge clk, așa cum se poate observa în Fig. 50. Un comportament similar este observat și la 820ns, unde qspo\_srst face tranziția de la 0 la 1. Totuși, acest semnal va avea efect doar la următorul **posedge clk**, adică la 850ns, moment în care qspo este resetat la 0, conform comportamentului resetului sincron.

#### **Reset Asincron**

```
// Reset asincron
resetAsync();
#13
readMemorySync(6'h30, readdata);
$display("Sync Read Data: %0h", readdata);
Fig. 89
```



La momentul 812ns, semnalul qspo\_rst face tranziția la 1, ceea ce activează resetul asincron. Spre deosebire de resetul sincron, care necesită un posedge clk pentru a avea efect, resetul asincron este aplicat imediat, fără a depinde de ceas. qspo\_rst devine 0 după 1ns, așa cum se poate observa în Fig. 51. Astfel, qspo devine 0 instantaneu, reflectând comportamentul așteptat al unui reset asincron. În această stare, qspo va rămâne 0 până când va apărea o nouă operație de scriere sau citire care să modifice conținutul său.

# Bibliografie

- 1. <a href="https://docs.amd.com/v/u/en-US/dist\_mem\_gen\_ds322">https://docs.amd.com/v/u/en-US/dist\_mem\_gen\_ds322</a>
- 2. <a href="https://www.chipverify.com/tutorials/systemverilog">https://www.chipverify.com/tutorials/systemverilog</a>
- 3. <a href="https://www.chipverify.com/tutorials/uvm">https://www.chipverify.com/tutorials/uvm</a>