

## Budapesti Műszaki és Gazdaságtudományi Egyetem

Villamosmérnöki és Informatikai Kar Méréstechnika és Információs Rendszerek Tanszék

# Rendszerarchitektúrák

Házi feladat

AXI - SPI perifériaillesztő

Kardos Bálint, ZI84PX Murányi Péter, A74MW9 Konzulens: Raikovich Tamás

# Tartalomjegyzék

| 1.          | Feladat                   | 1  |  |  |  |  |  |  |
|-------------|---------------------------|----|--|--|--|--|--|--|
| 2.          | AXI Lite                  | 1  |  |  |  |  |  |  |
| 3. SPI      |                           |    |  |  |  |  |  |  |
| 4. Tervezés |                           |    |  |  |  |  |  |  |
| 5.          | Verilog megvalósítás      | 1  |  |  |  |  |  |  |
|             | 5.1. AXI Lite illesztő    | 1  |  |  |  |  |  |  |
|             | 5.2. SPI modul            | 2  |  |  |  |  |  |  |
| 6.          | Szimuláció                | 2  |  |  |  |  |  |  |
|             | 6.1. BFM                  | 3  |  |  |  |  |  |  |
|             | 6.2. EEPROM               | 3  |  |  |  |  |  |  |
|             | 6.3. Szimuláció eredménye | 3  |  |  |  |  |  |  |
| 7.          | Forráskód                 | 6  |  |  |  |  |  |  |
|             | 7.1. SPI                  | 6  |  |  |  |  |  |  |
|             | 7.2. Busz illesztő        | 8  |  |  |  |  |  |  |
|             | 7.3. Szimulációs kód      | 15 |  |  |  |  |  |  |
| Hi          | ivatkozások               | 19 |  |  |  |  |  |  |

#### 1. Feladat

#### 2. AXI Lite

### 3. SPI

#### 4. Tervezés

blablabla

majd blokvázlatot ide

|   | 7              | 6 | 5 | 4 | 3 | 2             | 1     | 0       |  |
|---|----------------|---|---|---|---|---------------|-------|---------|--|
| 0 | [7:0] DATA I/O |   |   |   |   |               |       |         |  |
| 4 | X              |   |   |   |   | [2:0] CLK_DIV |       |         |  |
| 8 |                | X |   |   |   | 34            | IT EN | TX DONE |  |
| С | X              |   |   |   |   |               | CS    | BUSY    |  |

1. ábra. A periféria regiszterei

# 5. Verilog megvalósítás

#### 5.1. AXI Lite illesztő

Az AXI lite illesztő a Xilinx saját példája alapján lett megírva. Minden csatorna elkülöníthető, a funkciójukat önállóan valósítják meg. Mivel általában az ehhez hasonló rendszerek active-low reset-et használnak, így itt is ez lett alkalmazva. A reset és órajel vonalak minden egységhez csatlakoznak, a reset a rendszer minden elemét egy megadott indítási állapotba hozza.

Mivel regisztereket nem szabad egyszerre több helyről is írni ezért a bemeneti és kimeneti regisztereket el kell különíteni az AXI illesztőn belül. Egyes regiszterek értékét csak az AXI master változtatja, ezeket egyszerűen vissza lehet kötni a kimeneti regiszterekbe, viszont egy csomó regiszter más értéket ad ki olvasáskor mint amit beleírtunk, ezeknek a kezelésére külön logika szükséges.

Az írási csatornát kezelő logika végzi a bejövő adatok kezelését. Érvényes AWVALID jelre, ha éppen nem folyik írás, jelez az AWREADY vonalon hogy képes érvényes cím fogadására és mintavételezi a AWADDR csatornát. Ezután ugyanez a handshake folyamat megy végbe az írási adat csatornán is. Ha mindkét csatornán érvényes volt a tranzakció, megtörténik a tényleges írás a címzett adatregiszter AXI\_WSTRB által megadott megfelelő bájt vonalaiba. Ezzel együtt történik az írás válasz generálása a saját csatornáján, ami itt mindig nulla: írás OK. Mivel az eszköz úgy lett megtervezve hogy az adat I/O regiszterbe való írás indít egy új SPI tranzakciót, így szükség van arra hogy detektáljuk mikor történik írás a regiszterbe. Erre egy külön reg lett létrehozva ami 1-be állítódik amikor az AXI Lite nullás regiszterébe történik írás, 0-ba ha bármi más történik. Ezzel egy egy órajelű start jelet tudunk létrehozni amit az SPI modulnak

továbbítva megtörténhet az írás.

Az olvasási csatorna végzi a master felől jövő írási kérések kiszolgálását. Az olvasás mindössze két csatornát igényel. Érvényes olvasási handshake után a mintavételezett címnek megfelelő regiszterből megtörténik az adat kiírása a az olvasási válasz státusszal (itt mindig nulla) együtt. A kimenő regiszterek rendre mind wire adattípust használnak, ezek meghajtása vagy egy belső regiszterről történik aminek az értékét valamilyen logika határozza meg, vagy fixen nullára vannak bekötve (mivel 32 bitesek az AXI Lite regiszterek, míg legfeljebb is csak 8 bit van meghajtva egy regiszterben).

A regiszterek funkciójuktól függően különbözően viselkednek. Az I/O regiszter írási oldalról az SPI adatbemenetére kapcsolódik, míg olvasási oldalról az adatkimenetéhez van kötve. Ennek a regiszternek az írása egyben egy új SPI kommunikációt is elindít. Az órajelosztást állító regiszter a beírt értékét adja vissza olvasáskor, illetve közvetlenül csatlakozik az SPI clkdiv bemenetéhez. Az ITENABLE és TXDONE biteket tartalmazó regiszter ennél összetettebben működik. A TXDONE bit minden írás befejeződésekor magas állapotba kerül. Innen nullázható egy 1-es beírásával, illetve egy új SPI ciklus indításával ami automatikusan nullázza ezt az értéket. A negyedik regiszter a busy és  $\overline{CS}$  -t vezérlő biteket tartalmazza, előbbi csak olvasható, utóbbi írható és olvasható is.

Ezeken túl itt van elhelyezve az SPI-t indító logika, illetve a megszakítás generálása. Az SPI indítása egy egyszerű logikai és kapcsolat, ami az I/O regiszterbe történő írást jelző bitet, az SPI használatát jelölő "busy" bit negáltját és CS bit állását fogja össze. Ebből egy egy órajeles start impulzus jön létre ami elindítja az SPI modult. Az interrupt jel haszonló módon, adott bitek megfelelő állása esetén generálódik. Figyelembe veszi hogy aktiváltuk e az SPI TX\_ENABLE bitjét, és az SPI modul TXDONE kimeneti bitjét figyeli. Ha ez a kettő egyszerre igaz, egy egy órajeles megszakítást generál, jelölve hogy befejeződött az SPI kommunikáció.

#### 5.2. SPI modul

WOO állapotgépek !!!!4444négy

#### 6. Szimuláció

A létrehozott periféria működése szimulációval lett ellenőrizve. A szimulációhoz a Vivado beépített szimulátorát használtuk. A perifériához az AXI interfész felől egy AXI-LITE Master-t szimuláló BFM lett illesztve, míg az SPI oldalról egy Microchip EEPROM funkcionális verilog modellje lett illesztve. A szimuláció egy az EEPROM-ba való írást, majd onnan a beírt adatok kiolvasását valósítja meg.

A szimuláció 100MHz-s órajelet használ, a reset vonal induláskor 20ns ideig aktív alacsony állapotban van, majd visszatér magas szintre, ami elindítja az összes eszközt. Ezek után futnak le az AXI taszkok, amik inicializálják az SPI perifériát, beírnak 3 bájtot az illesztett EEPROM-ba, majd ezeket ugyanonnan kiolvassa. Az egész szimuláció 20us-ot vesz igénybe.

Az SPI modul /16 órejelosztást használ, amivel az SCK 6,25 MHz lesz. A SPI megszakítás küldője be lett kapcsolva ugyanis ezt használjuk az SPI tranzakciós ciklus befejeződésének figyelésére. A  $\overline{CS}$  jelet manuálisan állítjuk, az SPI megfelelő regiszterébe való írással.

#### 6.1. BFM

A busz funkcionális modell egyszerű verilog taszkokkal lett megvalósítva. Ezek a taszkok az AXI LITE interfészt ismertető részben bemutatott időzítési diagram szerint lettek megírva. A taszkokban az órajel felfutó éle után mindenhol egy 1ns-os késleltetés lett elhelyezve, hogy szimuláljuk a rendszerben a flip-flop-ok set-up és hold késleltetésüket. Az írási taszk két bemenettel rendelkezik, írási cím és írási adat. Az olvasási taszk egy bemenet és egy kimenettel rendelkezik: az olvasandó adat címe és a beolvasott adat. Verilog taszkok kimenetein szimulációban csak a taszk befejeződését követően jelenik meg az új adat, attól függetlenül hogy a taszkban hol történt ténylegesen a kimeneti regiszter írása.

#### 6.2. EEPROM

Az eszközhöz egy Microchip 25AA010A EEPROM szimulációs verilog modellje van illesztve, hogy a valós működés is tesztelve legyen. Ez a modul egy 1kbit méretű flash EEPROM-ot valósít meg, valós időzítési értékekkel. Az eszköz leírása ezen a linken érhető el.

Az eszköz valós időzítéseket is szimulál, ezért például az SCK órajel nem lehet gyorsabb 10MHz-nél, ugyanis az eszköz nem lesz képes elég gyorsan reagálni az órajel változására. Ezt az SPI órajelosztójának megfelelő beállításával lehet elérni.

Hogy az eszközt írhassuk, először egy instrukciót kell neki küldeni ami engedélyezi az írást az eszközön belül. Ehhez egy külön írási ciklusra van szükségünk ahol  $\overline{CS}$  jelet aktív alacsonyra kell húzni, majd az írásengedélyező (WREN) parancsot kiküldve a  $\overline{CS}$  jelet egy órajelciklus idejére vissza kell engedni magas szintre. Ha ezt nem tesszük meg, nem fogunk tudni írni az eszközbe. Ezek után lehet ténylegesen írni az eszközt. Íráskor először egy 8-bites írás instrukciót kell kiküldeni, ez után jöhet a cím, szintén 8-biten, majd ezt követően kezdődik meg a tényleges adat kiküldése. Lehet bájtos, illetve burst módban írni, viszont burst módban maximum 16 bájtot lehet kiküldeni, ugyanis az EEPROM csak ekkora bemeneti bufferrel rendelkezik. Az írást a  $\overline{CS}$  magasra húzásával lehet befejezni, ekkor az írásengedélyező belső flag automatikusan 0-ra áll vissza. Az EEPROM verilog modelljében apró változtatásokat kellett eszközölni, ugyanis a valóságot szimulálva 5ms-ig késleltet mielőtt az adatot ténylegesen beírja az adattároló regisztereibe. Ezt hogy a szimuláció rövidebb legyen, és a kapott hullámformák átláthatóbbak legyenek 100ns-re csökkentettük.

Olvasáshoz nem kell speciális parancsot megadni, elég az olvasás instrukció és olvasandó adat kezdőcíme, majd annyi bájtot tudunk kiolvasni amennyit akarunk, az EEPROM minden bájt után automatikusan növeli a pointer értékét. Olvasásnál nincs 16 bájtos limit, amíg el nem érjük a címtartomány végét, folyamatosan tudunk adatot kiolvasni. Az EEPROM verilog modellje a valóságot szimulálva, ha éppen nem küld adatot a MASTER felé a soros adatkimenet (SO) lábát magas impedanciás módba rakja (ez látható is a szimulációban).

#### 6.3. Szimuláció eredménye

Az ?? ábrán ábrán látható a teljes szimuláció eredménye. Láthatóak az AXI periféria írások és olvasások, illetve az SPI ciklusok. A beírt 3 adatbájt olvasáskor megjelenik a kimeneten, ebből látható hogy az SPI illesztő és az EEPROM is megfelelően működik. Ezeken felül a Vivado lehetőséget ad hogy a belső regiszterek állását is ellenőrizni lehessen, ha ezt megtesszük látható lesz hogy az EEPROM-nak valóban beíródott a belső regisztereibe az adat.



2. ábra. A teljes szimuláció

A ?? ábrán látható egy AXI írási ciklus. Látható hogy a cím és adatcsatorna egyszerre működtethető, mindkettő jelzi hogy új adat érkezik, majd vár a slave felől a válasz jelre hogy a handshake megtörténjen és az adat beíródjon a slave eszközbe. Az írásra a slave eszköz a write-response csatornán választ ad a megfelelő hibakóddal (esetünkben csak 0 lehet) és ezzel a



3. ábra. AXI írási ciklusok

master befejezi az írási ciklust. Mivel az AXI Lite nem támogat burst átvitelt így itt minden adatot egy komplett írási ciklussal kell átvinni.



4. ábra. SPI kommunikáció

Maga az SPI működése a ?? ábrán látható. Látszik hogy a feladatban megadott üzemmód szerint felfutó SCK jelre történik az adat mintavételezése, míg lefutó élre az új adat kiküldése. Látszik hogy amíg az EEPROM-nak nem küldünk írási utasítást a kimenete (MISO) magas impedanciás állapotban van. Látható hogy a TXCOMPLETE interrupt minden tranzakció végén egy óralejre megjelenik, ezzel jelezve a ciklus befejeződését. Érdekes megfigyelni hogy mivel az EEPROM valós időzítésekkel szimulál, az új adatot csak az SCK lefutása után jelentős idővel küldi ki.



5. ábra. AXI olvasási ciklusok

A ?? látható egy AXI olvasási ciklus. Látszik hogy a címet megadva és a handshake után megjelenik a kiolvasandó adat a megfelelő csatornáján.

## 7. Forráskód

#### 7.1. SPI

```
'timescale 1ns / 1ps
module SPI_MASTER( //Clock and reset
                    input clk,
                    input rstn,
                    //Registers
                    input [2:0] clk_div,
                    input [7:0] data_in,
                    output reg [7:0] data_out,
                    output reg busy,
                    output reg tx_complete,
                    input start_tx,
                    //IO
                    output reg sck,
                    output MOSI,
                    input MISO
   );
   //Clock divider
   reg [7:0] cntr = 0;
```

```
reg cntr_enable = 0;
always @(posedge clk)
begin
    #1; //Debug delay
    if((~rstn)||(~cntr_enable))
        cntr <= 0;
        cntr <= cntr + 1;
end
wire spi_tick = cntr[clk_div];
//Controll state machine
reg [1:0] state = 0;
reg [7:0] TX_BUFFER;
reg [7:0] RX_BUFFER;
reg spi_tick_prev;
reg [3:0] tx_count;
assign MOSI = TX_BUFFER[7];
//State machine
always @(posedge clk)
begin
    if (~rstn)
        begin
            state <= 0;
             TX_BUFFER <= 0;
            data_out <= 0;
            tx_complete <= 0;</pre>
             spi_tick_prev <= 0;
             tx_count <= 0;
            sck <= 0;
            cntr_enable <= 0;</pre>
        end
    else
        case (state)
            2'b00: //Idle state
                begin
                     #1; //Debug delay
                     tx_complete <= 0;</pre>
                     spi_tick_prev <= 0;
                     tx_count <= 0;
                     busy <= 0;
                     if(start_tx)
                         begin
                             state <= 2'b01;
                             busy <= 1;
                             TX_BUFFER <= data_in;
                             cntr_enable <= 1;</pre>
                             RX_BUFFER <= 0;
                         end
                 end
```

```
2'b01: //Transmit state
                     begin
                         #1; //Debug delay
                         //Shift data out falling edge
                         if ((spi_tick_prev) && (~spi_tick))
                             begin
                                 TX_BUFFER <= {TX_BUFFER[6:0],1'b0};
                                 sck <= 0;
                                 tx_count <= tx_count + 1;</pre>
                         //Sample data on rising edge
                         if((~spi_tick_prev) && (spi_tick) && (tx_count != 4'd8))
                             begin
                                 RX_BUFFER <= {RX_BUFFER[6:0], MISO};
                                  sck <= 1;
                         //Update tick
                         spi_tick_prev <= spi_tick;</pre>
                         //Terminate transfer
                         if(tx_count == 4, d8)
                             begin
                                  state <= 2'b10;
                                 sck <= 0;
                     end
                 2'b10: //TX done state
                     begin
                         #1; //Debug delay
                         tx_complete <= 1;</pre>
                         state <= 2, b00;
                         data_out <= RX_BUFFER;
                         TX_BUFFER <= 0;
                         cntr_enable <= 0;</pre>
                 default:
                     state <= 0;
            endcase
    end
endmodule
```

#### 7.2. Busz illesztő

```
'timescale 1ns / 1ps

//AXI Lite controller with 4 32 bit registers

module AXI_LITE_CNTRL(

//General

input wire S_AXI_ACLK, //Clock

input wire S_AXI_ARESETN, //Active LOW reset

//AXI4-Lite wires

//Write address channel
```

```
input wire [3:0] S_AXI_AWADDR, //Address, 4 bits
                  input wire S_AXI_AWVALID, //Write address valid
                  output reg S_AXI_AWREADY, //Write address ready
                 //Write data channel
                 input wire [31:0] S_AXI_WDATA, //Input data
                 output reg S_AXI_WREADY, //Ready for write
                 //Write response channel
                 output reg [1:0] S_AXI_BRESP, //Write response
                                            // Write response
                 output reg S_AXI_BVALID,
                 input wire S_AXI_BREADY,
                                                    //Response ready
                  //Read address channel
                 input wire [3:0] S_AXI_ARADDR, //Read address
                 input wire S_AXI_ARVALID,
                                                    //Read address valid
                 output reg S_AXI_ARREADY,
                                                    //Read address ready
                 //Read data channel
                 output reg [31:0] S_AXI_RDATA, //Read data
                 output reg [1:0] S_AXI_RRESP, //Reade response
                 output reg S_AXI_RVALID,
                                                   //Read valid
                 input wire S_AXI_RREADY,
                                                   //Read ready
                 //SPI signals
                 output CS,
                 output MOSI,
                 output SCK,
                 input MISO,
                 //Interrupt
                 output reg TX_DONE_IT
);
reg aw_enable = 0;
   //Write address latching
reg [3:0] axi_waddr = 0;
always @ (posedge S_AXI_ACLK)
  if (~S_AXI_ARESETN)
       begin
         axi_waddr <= 0;
       end
   else
         if(~S_AXI_AWREADY && S_AXI_AWVALID && S_AXI_WVALID && aw_enable)
              axi_waddr <= S_AXI_AWADDR;
       end
end
//Awready gen
always @ (posedge S_AXI_ACLK)
begin
   if (~S_AXI_ARESETN) //Reset signal
      begin
          S_AXI_AWREADY <= O;
```

```
aw_enable <= 1;
        end
    else
            if(~S_AXI_AWREADY && S_AXI_AWVALID && S_AXI_WVALID && aw_enable)
                begin
                   S_AXI_AWREADY <= 1;
                   aw_enable <= 0;
            else if(S_AXI_BREADY && S_AXI_BVALID)
                  S_AXI_AWREADY <= O;
                   aw_enable <= 1;
            else
               S_AXI_AWREADY <= 0;
        end
end
//Wready gen
always @ (posedge S_AXI_ACLK)
begin
   if (~S_AXI_ARESETN)
       begin
           S_AXI_WREADY <= 0;
    else
        begin
           if(~S_AXI_WREADY && S_AXI_WVALID && S_AXI_AWVALID && aw_enable)
               begin
                  S_AXI_WREADY <= 1;
                end
                begin
                   S_AXI_WREADY <= 0;
       end
end
//Write logic
//---INPUT REGISTERS-----
reg [31:0] input_reg0 = 0;
reg [31:0] input_reg1 = 0;
reg [31:0] input_reg2 = 0;
reg [31:0] input_reg3 = 0;
//New TX trigger signal
reg NEW_TX = 0;
reg TX_DONE_CLEAR;
always @ (posedge S_AXI_ACLK)
begin
   if (~S_AXI_ARESETN)
    begin
```

```
input_reg0 <= 0;
        input_reg1 <= 0;
        input_reg2 <= 0;
        input_reg3 <= 0;
    end
else
    begin
        if(S_AXI_WREADY && S_AXI_WVALID && S_AXI_AWREADY && S_AXI_AWVALID)
                case(axi_waddr[3:2])
                    2'h0:
                        begin
                             if (S_AXI_WSTRB[0] == 1)
                                 begin
                                     input_reg0[7:0] <= S_AXI_WDATA[7:0];
                                     NEW_TX <= 1;
                             if (S_AXI_WSTRB[1] == 1)
                                 input_reg0[15:8] <= S_AXI_WDATA[15:8];
                             if (S_AXI_WSTRB[2] == 1)
                                 input_reg0 [23:16] <= S_AXI_WDATA [23:16];
                             if(S_AXI_WSTRB[3] == 1)
                                 input_reg0[31:24] <= S_AXI_WDATA[31:24];
                    2'h1:
                        begin
                             if (S_AXI_WSTRB[0] == 1)
                                     input_reg1[7:0] <= S_AXI_WDATA[7:0];
                             if (S_AXI_WSTRB[1] == 1)
                                 input_reg1[15:8] <= S_AXI_WDATA[15:8];
                             if (S_AXI_WSTRB[2] == 1)
                                 input_reg1[23:16] <= S_AXI_WDATA[23:16];
                             if (S_AXI_WSTRB[3] == 1)
                                 input_reg1[31:24] <= S_AXI_WDATA[31:24];
                         \tt end
                    2'h2:
                         begin
                             if (S_AXI_WSTRB[0] == 1)
                                 begin
                                     input_reg2[7:0] <= S_AXI_WDATA[7:0];
                                     TX_DONE_CLEAR <= 1;
                             if (S_AXI_WSTRB[1] == 1)
                                 input_reg2 [15:8] <= S_AXI_WDATA [15:8];
                             if (S_AXI_WSTRB[2] == 1)
                                 input_reg2 [23:16] <= S_AXI_WDATA [23:16];
                             if (S_AXI_WSTRB[3] == 1)
                                 input_reg2 [31:24] <= S_AXI_WDATA [31:24];
                         end
                    2'h3:
                         begin
                             if (S_AXI_WSTRB[0] == 1)
                                 input_reg3 [7:0] <= S_AXI_WDATA [7:0];
                            if (S_AXI_WSTRB[1] == 1)
```

```
input_reg3 [15:8] <= S_AXI_WDATA [15:8];
                                if (S_AXI_WSTRB[2] == 1)
                                    input_reg3 [23:16] <= S_AXI_WDATA [23:16];
                                if (S_AXI_WSTRB[3] == 1)
                                   input_reg3 [31:24] <= S_AXI_WDATA [31:24];
                        default:
                           begin
                            end
                    endcase
           else
                begin
                   NEW_TX <= 0;
                   TX_DONE_CLEAR <= 0;
       end
end
//Write response-----
always @ (posedge S_AXI_ACLK)
   if (~S_AXI_ARESETN)
           S_AXI_BVALID <= 0;
           S_AXI_BRESP <= 0;
        end
    else
           if(S_AXI_AWREADY && S_AXI_AWVALID && ~S_AXI_BVALID && S_AXI_WREADY && S_AXI_WVALID)
               begin
                   S_AXI_BVALID <= 1;
                   S_AXI_BRESP <= 0;
            else
                  if(S_AXI_BREADY && S_AXI_BVALID)
                       S_AXI_BVALID <= 0;
               end
       end
end
//Implement axi_arready generation
reg [3:0] axi_araddr = 0;
always @ (posedge S_AXI_ACLK)
   if (~S_AXI_ARESETN)
            S_AXI_ARREADY <= 0;
           axi_araddr <= 0;
        end
    else
```

```
begin
            if (~S_AXI_ARREADY && S_AXI_ARVALID)
                   S_AXI_ARREADY <= 1;
                   axi_araddr <= S_AXI_ARADDR;
            else
                begin
                  S_AXI_ARREADY <= 0;
       end
end
//Implement axi_arvalid generation
always @ (posedge S_AXI_ACLK)
begin
   if (~S_AXI_ARESETN)
       begin
            S_AXI_RVALID <= 0;
            S_AXI_RRESP <= 0;
    else
        begin
            if(S_AXI_ARREADY && S_AXI_ARVALID && ~S_AXI_RVALID)
               begin
                   S_AXI_RVALID <= 1;
                   S_AXI_RRESP <= 0;
            else if(S_AXI_RVALID && S_AXI_RREADY)
                  S_AXI_RVALID <= 0;
       end
end
//Read logic
//----Output registers-----
wire [31:0] output_reg0;
wire [31:0] output_reg1;
wire [31:0] output_reg2;
wire [31:0] output_reg3;
always @ (posedge S_AXI_ACLK)
begin
   if (~S_AXI_ARESETN)
       begin
          S_AXI_RDATA <= 0;
        end
    else
        begin
           if (S_AXI_ARREADY && S_AXI_ARVALID && ~S_AXI_RVALID)
                   case(axi_araddr[3:2])
```

```
2'h0 : S_AXI_RDATA <= output_reg0;
                        2'h1 : S_AXI_RDATA <= output_reg1;
                        2'h2 : S_AXI_RDATA <= output_reg2;
                        2,h3 : S_AXI_RDATA <= output_reg3;
                    endcase
                end
       end
end
//SPI controll logic
//Data register IN trigger
wire SPI_busy;
wire TX_START = ~SPI_busy & NEW_TX & (~CS);
//IT generation & TX_COMPLETE flag
reg TX_DONE_REG;
wire TX_DONE_SPI;
assign output_reg2[0] = TX_DONE_REG;
assign output_reg2[1] = input_reg2[1];
always @ (posedge S_AXI_ACLK)
begin
    if (~S_AXI_ARESETN)
        begin
           TX_DONE_REG <= 0;
            TX_DONE_IT <= 0;
        end
    else
        begin
            if(TX_DONE_SPI)
                begin
                    TX_DONE_REG <= 1;
                    if(input_reg2[1])
                       TX_DONE_IT <= 1;
            else if(((TX_DONE_CLEAR) && (input_reg2[0])) || (TX_START))
                   TX_DONE_REG <= 0;
                    TX_DONE_IT <= 0;
            else
                TX_DONE_IT <= 0;
        end
end
//REG4 CS BUSY
assign CS = ~input_reg3[1];
assign output_reg3[1] = input_reg3[1];
assign output_reg3[0] = SPI_busy;
//SPI connection
```

```
SPI_MASTER SPI( .clk(S_AXI_ACLK),
                   .rstn(S_AXI_ARESETN),
                    .clk_div(input_reg1[2:0]),
                   .data_in(input_reg0[7:0]),
                   .data_out(output_reg0[7:0]),
                   .busy(SPI_busy),
                   .tx_complete(TX_DONE_SPI),
                   .start_tx(TX_START),
                   .MOSI(MOSI),
                   .MISO(MISO),
                   .sck(SCK)
                   );
   //Etc
   assign output_reg0[31:8] = 24'h0;
   assign output_reg1 = {29'h0,input_reg1[2:0]};
   assign output_reg2[31:2] = 30'h0;
   assign output_reg3[31:2] = 30'h0;
endmodule
```

#### 7.3. Szimulációs kód

```
'timescale 1ns / 1ps
module EEPROM_SPI_TB(
  );
   //Clock generation
   reg S_AXI_ACLK = 0;
   always
   begin
       #5 S_AXI_ACLK <= ~S_AXI_ACLK;
   //Reset signal generation
   reg S_AXI_ARESETN = 0; //Initial state
   always
   begin
       #20;
       @(posedge S_AXI_ACLK);
       #1 S_AXI_ARESETN <= 1;
   //Connections
   //AXI4-Lite wires
   //Write address channel
   reg [3:0] S_AXI_AWADDR = 0;
   reg S_AXI_AWVALID = 0;
   wire S_AXI_AWREADY;
```

```
//Write data channel
reg [31:0] S_AXI_WDATA = 0;
reg [3:0] S_AXI_WSTRB = 0;
reg S_AXI_WVALID = 0;
wire S_AXI_WREADY;
//Write response channel
wire [1:0] S_AXI_BRESP;
wire S_AXI_BVALID;
reg S_AXI_BREADY = 0;
//Read address channel
reg [3:0] S_AXI_ARADDR = 0;
reg S_AXI_ARVALID = O;
wire S_AXI_ARREADY;
//Read data channel
wire [31:0] S_AXI_RDATA;
wire [1:0] S_AXI_RRESP;
wire S_AXI_RVALID;
reg S_AXI_RREADY = 0;
//SPI IO
wire MOSI;
wire MISO;
wire SCK;
wire TX_DONE_IT;
wire CS;
//AXI Lite transaction tasks
//Write transaction
task axi_write;
    input [3:0] address;
    input [31:0] data;
    begin
        @(posedge S_AXI_ACLK);
        #1;
        S_AXI_AWADDR <= address;
        S_AXI_AWVALID <= 1;
        S_AXI_WDATA <= data;
        S_AXI_WSTRB <= 4'b1111;
        S_AXI_WVALID <= 1;
        S_AXI_BREADY <= 1;
        wait (S_AXI_AWREADY);
        wait(S_AXI_WREADY);
        @(posedge S_AXI_ACLK);
        S_AXI_WVALID <= 0;
        S_AXI_AWVALID <= 0;
        wait(S_AXI_BVALID);
        @(posedge S_AXI_ACLK);
        S_AXI_BREADY <= 0;
    end
endtask
//Read transaction
task axi_read;
    input [3:0] address;
    output reg [31:0] data;
```

```
begin
       @(posedge S_AXI_ACLK);
       #1;
       S_AXI_ARADDR <= address;</pre>
       S_AXI_ARVALID <= 1;
       S_AXI_RREADY <= 1;
       wait (S_AXI_ARREADY);
       @(posedge S_AXI_ACLK);
       S_AXI_ARVALID <= 0;
       wait (S_AXI_RVALID);
       @(posedge S_AXI_ACLK);
       data <= S_AXI_RDATA;
       #1;
       S_AXI_RREADY <= 0;
   end
endtask
//Read buffer
reg [31:0] read_data = 0;
//Test cases
always
begin
   #40;
   //Set up SPI
   axi_write(4,3); //CLK div /16
   //Pull CS low
    axi_write(12,8'b10);
   //WRITE LATCH ON
   axi_write(0,8'b00000110);
   //CS high
   wait(TX_DONE_IT);
   axi_write(12,8'b00);
   #50;
   //CS low
   axi_write(12,8'b10);
   //Write data
    axi_write(0,8'b00000010); wait(TX_DONE_IT);//Write instruction
    axi_write(0,8'b00000010); wait(TX_DONE_IT); //ADDRESS
    axi_write(0,8'hAA); wait(TX_DONE_IT); //DATA
    axi_write(0,8'hBB); wait(TX_DONE_IT);
    axi_write(0,8'hC5); wait(TX_DONE_IT);
    axi_write(12,8'b00); //CS high
    #500;
    //Read data
    axi_write(12,8'b10); //CS low
    axi\_write(0,8'b00000011); wait(TX\_DONE\_IT);//Read instruction
    axi_write(0,8'b0); wait(TX_DONE_IT); //Start transfer
    axi_read(0,read_data); //READ\hat{u}
    axi_write(0,8'b0); wait(TX_DONE_IT); //Start transfer
    axi_read(0,read_data); //READ
    axi_write(0,8',00); wait(TX_DONE_IT); //Start transfer
```

```
axi_read(0,read_data); //READ
        axi_write(12,8'b00); //CS high
        #5000000;
   end
   //Connect to UUT
   AXI_LITE_CNTRL UUT( .S_AXI_ACLK(S_AXI_ACLK),
                        .S_AXI_ARESETN(S_AXI_ARESETN),
                        .S_AXI_AWADDR(S_AXI_AWADDR),
                        .S_AXI_AWVALID(S_AXI_AWVALID),
                        .S_AXI_AWREADY(S_AXI_AWREADY),
                        .S_AXI_WDATA(S_AXI_WDATA),
                        .S_AXI_WSTRB(S_AXI_WSTRB),
                        .S_AXI_WVALID(S_AXI_WVALID),
                        .S_AXI_WREADY(S_AXI_WREADY),
                        .S_AXI_BRESP(S_AXI_BRESP),
                        .S_AXI_BVALID(S_AXI_BVALID),
                        .S_AXI_BRE ADY (S_AXI_BRE ADY),
                        .S_AXI_ARADDR(S_AXI_ARADDR),
                        .S_AXI_ARVALID(S_AXI_ARVALID),
                        .S_AXI_ARREADY(S_AXI_ARREADY),
                        .S_AXI_RDATA(S_AXI_RDATA),
                        .S_AXI_RRESP(S_AXI_RRESP),
                        .S_AXI_RVALID(S_AXI_RVALID),
                        .S_AXI_RREADY(S_AXI_RREADY),
                        .MOSI(MOSI),
                        .MISO(MISO),
                        .SCK(SCK),
                        .TX_DONE_IT(TX_DONE_IT),
                        .CS(CS)
                        );
   //EEPROM
   M25AAO1OA EEPROM(
                       .SI(MOSI),
                        .SCK(SCK),
                        .CS_N(CS),
                        .WP_N(1'b1),
                        .HOLD_N(1'b1),
                        .RESET (~S_AXI_ARESETN),
                        .SO(MISO));
endmodule
```

# Hivatkozások

```
[1] SPI leírása https://en.wikipedia.org/wiki/Serial_Peripheral_Interface_Bus 2018. május 16.
[2] AXI Lite leírása és időzítési diagramja https://www.xilinx.com/support/documentation/ip_documentation/axi_lite_ipif/v3_0/pg155-axi-lite-ipif.pdf 2018. május 16.
[3] EEPROM leírása http://ww1.microchip.com/downloads/en/DeviceDoc/21832H.pdf 2018. május 16.
[4] 2018. május 16.
[5]
[6] 2018. május 16.
```