#### DIPLOMSKI RAD

# $Diplomski\ rad$

# Komunikacija između Linux operativnog sistema i hardvera realizovanog u FPGA

Dejan Lukić 82/2014

Predmetni profesor:

Lazar Saranovac

Predmetni asistenti:

Strahinja Janković

Elektrotehnički fakultet Univerzitet u Beogradu Letnji semestar 2017/2018

# Contents

| 1 | $\mathbf{U}\mathbf{v}\mathbf{o}$   | m od                                                          | 3  |  |
|---|------------------------------------|---------------------------------------------------------------|----|--|
|   | 1.1                                | Sistemi na čipu sa FPGA                                       | 3  |  |
|   | 1.2                                | Opis DE1-SoC                                                  | 3  |  |
|   | 1.3                                | Opis Altera Cyclone 5                                         | 4  |  |
|   |                                    | 1.3.1 Konfigurisanje FPGA i pokretanje HPS                    | 4  |  |
|   |                                    | 1.3.2 HPS-FPGA interfejsi                                     | 4  |  |
|   |                                    |                                                               | 5  |  |
|   | 1.4                                | Alati                                                         | 5  |  |
| 2 | Opis projektovanog sistema 7       |                                                               |    |  |
|   | $2.\overline{1}$                   | Memorijski mapiran interfejs ka LE diodama i tasterima u FPGA | 7  |  |
|   | 2.2                                |                                                               | 7  |  |
|   | 2.3                                | Bootloader                                                    | 7  |  |
|   | 2.4                                | Linuks kernel i Root File System                              | 8  |  |
|   | 2.5                                | Drajver                                                       | 8  |  |
| 3 | Uputstvo za repliciranje rezultata |                                                               |    |  |
|   | 3.1                                |                                                               | 9  |  |
|   | 3.2                                | · · · · · ·                                                   | 15 |  |
|   | 3.3                                |                                                               | 15 |  |
|   | 3.4                                |                                                               | 15 |  |
|   | 3.5                                |                                                               | 16 |  |
|   | 3.6                                |                                                               | 16 |  |
|   | 3.7                                | Linuks kernel                                                 | 18 |  |
|   | 3.8                                | Kompajliranje drajvera                                        | 19 |  |
|   | 3.9                                |                                                               | 19 |  |
|   | 3.10                               | ·                                                             | 20 |  |
|   | 3.11                               | Testiranje sistema                                            | 21 |  |
| 4 | Zak                                | ljučak 2                                                      | 23 |  |
| 5 | Doc                                | latak 2                                                       | 24 |  |
| • | 5.1                                |                                                               | 24 |  |
|   | -                                  | v                                                             | 00 |  |

#### 1 Uvod

U ovom radu je na DE1-SoC razvojnom sistemu implementiran jednostavan hardver u FPGA, portovan je Linux operativni sistem i napisan je drajver za pristup registrima i prihvatanje prekida iz FPGA.

## 1.1 Sistemi na čipu sa FPGA

Sa sve većim mogućnostima namenskih sistema došlo je do popularizacije sistema na čipovima (SoC - System on Chip) koji integrišu mikroprocesore sa više jezgara, memorije na čipu, mnogobrojne periferije i transivere, kao i FPGA (Field Programmable Gate Array).

Ova tehnologija daje dizajneru sistema veliku slobodu i mogućnosti, a zadršava se klasičan postupak projektovanja namenskih sistema. Uz to se ostvaruje veća integracija, manja potrošnja, manja površina štampane ploče (PCB - Printed Circuit Board) i veći protok podataka između procesora i FPGA dela.

Uobičajena primena ovih sistema je implementacija specifičnih akceleratora koji ubrzavaju izvršavanje algoritama i implementacija specifičnih programabilnih interfejsa ka spoljnom svetu. Nove tehnologije kao što su OpenCL, Vivado HLS, Matlab HDL Coder omogućavaju kompatibilnost dizajna softvera na visokom nivou i implementiranog hardvera na niskom nivou.

SoCFPGA sistemi najčešće sadrže ARM mikroprocesor. Aplikacije na mikroprocesoru bez operativnog sistema (baremetal) nude jednostavno pisanje koda i uštedu na resursima. Za kompleksnije aplikacije koriste se operativni sistemi (OS) i time se olakšava integrisanje mrežnih protokola, rad sa multimedijalnim sadržajima, kriptografskim bibliotekama kao i mnoge druge mogućnosti koje su dostupne kao open-source softver. Kada je potrebno garantovati reakciju u određenom vremenu na neki spoljni događaj veliki operativni sistemi nisu dobro rešenje i koriste operativni sistemi u realnom vremenu (RTOS).

Hardver u FPGA se projektuje upotrebom nekog dva popularna jezika za opis hardvera (Verilog i VHDL - Very High Speed Integrated Circuit Hardwer Description Language) i softverskih alata za specifični uređaj. Dodatno ovi alati olakšavaju dizajn upotrebom IP( Intelectual Property) blokova kao i generisanjem raznih izlaznih fajlova koji opisuju projektovani hardver na standardni način i koriste se prilikom razvoja softvera.

# 1.2 Opis DE1-SoC

U ovom radu korišćen je DE1-SoC razvojni sistem koji se vrlo često upotrebljava u edukativne svrhe. Razvojni sistem je zasnovan na čipu iz familije Cyclone V kompanije Intel (ranije Altera).

U nastavku su navedene samo osobine razvojnog sistema koje se tiču ovog rada, a detaljniji opis se moze pronaći u dokumentu DE1-SoC User Manual [] dodati referencu: https://www.terasic.com. SoC User Manual(rev.E Board) Terasic

- Sistem na čipu Cyclone V SoC 5CSEMA5F31
- Memorija 1GB (2x256Mx16) DDR3 SDRAM povezana na HPS
- Slot za Micro SD karticu povezan na HPS
- UART na USB (USB Mini-B konektor)
- 5 debaunsiranih tastera (FPGA x4, HPS x1)
- 11 LE dioda (FPGA x10, HPS x 1)
- 12V DC napajanje

## 1.3 Opis Altera Cyclone 5

Altera Cyclone 5 je SoC FPGA koji se sastoji od dva dela(slika): procesorski deo (HPS - Hard processor System) i programabilni FGPA deo. HPS se sastoji od MPU (Microprocessor unit) sa ARM Cortex-A9 MPCore sa dva jezgra i sledećih modula: kontroleri memorije, memorije, periferije, sistem interkonekcije, debug moduli, PLL moduli. FPGA deo se sastoji od sledećih delova: FPGA programabilna logika (look-up tabele, RAM memorije, mnozači i rutiranje), kontrolni blok, PLL, kontroler memorije.

Svaki pin kućista je povezan na samo jedan od ova dva dela sistema, tako da HPS deo i FPGA deo ne mogu međusobno razmenjivati pinove.

#### 1.3.1 Konfigurisanje FPGA i pokretanje HPS

Pri pokretanju HPS (boot) moze da učita program iz FPGA dela, iz eksterne flash memorije ili preko JTAG. FPGA ima mogućnost da se konfiguriše softverski iz HPS korišćenjem periferije FPGA Manager ili spoljnim programatorom. Kombinacije ovih mogucnosti daju nekoliko scenarija:

- nezavisno konfigurisanje FPGA i pokretanje HPS
- konfigurisanje FPGA, zatim pokretanje HPS iz memorije koja se nalazi u FPGA
- pokretanje HPS, zatim konfigurisanje FPGA iz HPS

DE1-SoC razvojni sistem dolazi sa integrisanim programatorom kojem se pristupa preko USB porta. Moguće je podesiti konfigurisanje FPGA spolja ili iz HPS upotrebom prekidača MSEL, dok se HPS uvek pokreće iz flash memorije SD kartice. (dodati tableu 3-2 iz de1soc user guide)

#### 1.3.2 HPS-FPGA interfejsi

HPS-FPGA interfejsi su komunikacioni kanali između HPS i FPGA dela. U nastavku su nabrojani i opisani HPS-FPGA interfejsi:

- FPGA-to-HPS bridge magistrala visokih preformansi konfigurabilne sirine od 32,64 ili 128 bita. Na ovoj magistrali je FPGA master. Ovaj interfejs otkriva FPGA masterima ceo adresni prostor HPS dela.
- HPS-to-FPGA bridge magistrala visokih preformansi konfigurabilne sirine od 32,64 ili 128 bita. Na ovoj magistrali je HPS master a u FPGA se nalazi slave.
- Lightweight HPS-to-FPGA magistrala sirine 32 bita. HPS je master na ovoj magistrali. Ovaj interfejs manjeg protoka je namenjen za pristup statusnim i kontrolnim registrima periferijama implementiranim u FPGA delu.
- FPGA manager HPS periferja koja komunicira sa FPGA delom prilikom konfiguracije ili pokretanja (boot)
- Prekidi mogucnost povezivanja prekida iz FPGA na HPS kontroler prekida
- HPS debug interfejs omogućava da se debug mogućnosti prošire i na FPGA deo

Interfejsi koji su produžetak AXI magistrale na FPGA deo su FPGA-to-HPS bridge, HPS-to-FPGA bridge i Lightweight HPS-to-FPGA. Za povezivanje na ovu magistralu sa strane FPGA koristi se Avalon magistrala, stoga je neophodan AXI-Avalon bridge.

#### 1.3.3 Proces pokretanja HPS (boot)

Pokretanje HPS je proces koji se obavlja u više koraka. Nakon izvršavanja svakog koraka se učitava i pokreće sledeći. Ovo je proces je sličan kod svih ARM procesora, a u nastavku je ukratko opisan za konkretnu platformu.



Figure 1: Tok pokretana sistema

Pri izlazu iz reset stanja procesor počinje izvrsavanje sa reset vektora iz memorije na čipu. Na adresi reset vektora je upisan Boot ROM progtam. Ovo je prvi korak u pokretanju HPS. BootROM izvršava osnovna podešavanja procesora i dohvata Preloader iz NOR flash memorije, NAND flah memorije ili SD/MMC flash memorije. Očitavaju se BSEL pinovi na osnovu kojih se određuje gde je smešten Preloader, zatim se inicijalizuje taj interfejs i učitava i pokreće Preloader. Boot ROM softver proizvođača i ne može se menjati.

Preloader je prvi korak u pokretanju koji može da se konfiguriše. Preloader koji se koristi u ovom radu je zasnovan na SPL (Secondary Program Loader) framework koji je deo U-Boot projekta, što znači da Preloader i U-Boot dele dosta izvornog koda, kao što je mnoštvo pouzdanih drajvera. Preloader obično izvršava inicijalizaciju SDRAM, dodatna podešavanja sitema, inicijalizaciju flash kontrolera koji sadrži sledeći program (NAND, SD/MMC, QSPI) i zatim učitavanje programa u RAM memoriju i pokretanje.

Softver koji sledi nakon Preloader-a može biti baremetal aplikacija ili bootloader. Preloader i svi prethodni programi se izvršavaju na prvom jezgru procesora dok je drugo u reset stanju. Naredni koraci mogu inicijalizovati drugo jezgro.

Bootloader ima zadatak da podesi promenljive okruženja operativnog sistema, dohvati fajlove za pokretanje operativni sistem (sa flash memorije, putem Etherneta preko TFTP protokola ili USB), konfigurise FPGA pruži konzolu za korisničke operacije. Neki od populatnih open-source bootloadera su U-Boot i Barebox.

#### 1.4 Alati

U nastavku će ukratko biti opisani korišćeni alati sa izdvojenim najvažnijim mogućnostima:

- Quartus Prime 18.0 alat za razvoj hardvera na FPGA. Deo paketa je Platform Designer (ranije Qsys) koji u dizajn ukljucuje HPS, IP blokove i definiše povezanost ovih delova
- Preloader Generator (bsp-editor alat iz SoC EDS) Generise izvorni kod Preloader-a na osnovi izlaznih fajlova koji opisuju hardver
- Device Tree Generator (sopc2dts alat iz SoC EDS) Generise Device Tree opis hardvera na osnovi izlaznih fajlova koji opisuju hardver
- DE1-SoC Builder Generise prazan Quartus projekat za DE1-SoC razvojni sistem
- Linaro Toolchain koristi se za kompajliranje softvera

Na slici 2 je grafički prikazan tok projektovanja jednog SoCFPGA sistema. U nastavku su objašnjeni fajlovi koji se koriste pri projektovanju:

- .qpf projektni fajl za Quartus. Ovaj fajl generiše DE1-SoC Builder
- .qsf skripta za podešavanje pinova. Ovaj fajl generiše DE1-SoC Builder



Figure 2: Tok projektovanja

- .sdc skripta za podešavanje takta. Ovaj fajl generiše DE1-SoC Builder
- .v Verilog HDL izvorni kod
- .vhd VHDL izvorni kod
- .sof SDRAM Object File fajl za programiranje FPGA. Ovaj fajl generiše Quartus pri kompajliranju dizajna
- .rbf Raw Binary File fajl za programiranje FPGA. Ovaj fajl se dobija konverzijom .sof alatom quartus\_cpf
- .dts Device Tree Source opis hardvera za Linuks kernel
- .dtb Device Tree Blob binarni fajl, kompajlirani opis hardvera za Linuks kernel
- .sopcinfo sadrži opis hardvera na osnovu kog se generišu drugi fajlovi. Ovaj fajl generiše Platform Designer
- .c izvorni kod u jeziku C
- Makefile sadrži set direktiva za make buld system

# 2 Opis projektovanog sistema

U ovom radu je implementiran jednostavan sistem koji demonstira osnovne mogucnosti u dizajniranju sistema na SoC FPGA. Na slici 7 je prikazan realizovani sistem (dodati novu sliku?)



Figure 3: Povezivanje keys\_0 bloka

## 2.1 Memorijski mapiran interfejs ka LE diodama i tasterima u FPGA

U FPGA delu su postavljena dva PIO (Parallel I/O) Intel FPGA IP bloka. PIO IP blok leds\_0 je izlazni i koristi se za kontrolisanje LE dioda. PIO IP blok keys\_0 je ulazni i koristi se za očitavanje tastera. PIO IP blok keys\_0 takođe šalje prekidni zahtev HPS-u na svaku uzlaznu i silaznu ivicu.

PIO blokovi keys\_0 i leds\_0 su povezani kao Memory Mapped Slave na Avalon magistralu. Prilikom dizajna se automatski generiše AXI-Avalon bridge i tako su PIO blokovi keys\_0 i leds\_0 memorijski mapirani u adresni prostor HPS-a.

#### 2.2 Preloader

Za pokretanje HPS sistema koristi se Preloader generisan Alterinim alatima na osnovu generisanih fajlova za opis harvera.

#### 2.3 Bootloader

Uobicajeni izbor za bootloader je U-Boot. U-Boot učitava konfiguracioni fajl u .rbf formatu i konfigurise FPGA. Zatim učitava binarni fajl za opis hardverske pratforme (.dtb fajl) i binarni fajl kernela operativnog sistema (zImage fajl). Argumenti za podešavanje sistena se prosleđuju pri prepustanju toka izvršavanja kernelu.

## 2.4 Linuks kernel i Root File System

Izvorni kod za Linuks kernel je preuzet za Alterinog git-a. Root File System je generisan korišćenjem buildroot-a i upisan na SD karticu.

## 2.5 Drajver

Drajver je napisan kao modul kernela koji se uključuje odgovarajućom komandom. Nakon uključivanja drajver iz binarnog opisa harvdera (Device Tree Blob) učitava informacije o harvderu. Drajver pravi fajlove u sysfs fajl sistemu. Čitanje i upis u ove fajlove poziva odgovarajuće funkcije u drajveru koje upravljaju hardverom.

Sistemski fajlovi koje pravi drajver i njihov opis:

- leds upisani broj se prikazuje na LE diodama u binarnoj predstavi
- keys čitanje vraća binarnu predstavu stanja tastera
- irq\_flag pristup registru za flegove prekida (1 na n-tom bitu označava pristigli prekid na n-tom tasteru, upis 1 na n-ti bit čisti n-ti fleg)
- irq\_mask čitanje i upis u registar za maksiranje prekida (upis 1 na n-ti bit omogućava prekid na n-tom tasteru)

Ceo kod drajvera je dostupan u dodatku.

## Testiranje sistema

Binarni fajlovi za pokretanje sistema i root fajl sistem su napisani na SD karticu. HPS je povezan sa PC računarom preko UART-USB serijske veze. Otvaranjem konzole na PC računaru se pristupa sistemu.

U nastavku je dat komplatan spisak koraka za realizovanje sistema.

# 3 Uputstvo za repliciranje rezultata

## 3.1 Projektovanje u Quartus-u

- 1. U Microsoft Windows operativnom sistemu pokrenuti DE1SoC\_SystemBuilder.exe (dostupan na [])
- 2. Izabrati konfiguraciju kao na slici 4 i izabrati Generate



Figure 4: Podešavanja DE1\_Soc Bulder-a

- 3. Izbristati ld\_dipl.v fajl (ovaj fajl se ne koristi, kasnije će biti napravljen ld\_dipl.vhd fajl za VHDL)
- 4. Kopirati generisane fajlove u Ubuntu u radni folder (u ovom radu je to ~/ld\_dipl/hw/)
- 5. Pokrenuti Quartus (Quartus Prime 18.0) Lite Edition
- 6. Otvoriti projekat komandom File > Open Project... i izabrati ~/ld\_dipl/hw/ld\_dipl.qpf
- 7. U prozoru Tasks izabrati Edit Settings, u novom prozoru pod Timing Analyser ispod teksta SDC files to include in the project klikom na dugme '...' izabrati fajl ld\_dipl.sdc
- 8. Pokrenuti Platform Designer klikom na Tools > Platform Designer
- Iz prozora IP Catalog izabrati Processors and Peripherials > Hard Processor Systems
   ArriaV/Cyclone V Hard Processor System. Ovim se otvara meni za podešavanje
   HPS modula
- 10. Pod tabom FPGA interfaces izvršiti sledeće izmene:
  - U opštim podešavanjima isključiti opciju Enable MPU standby and event signals

- U podešavanjima AXI Bridges podesiti FPGA-to-HPS interface i HPS-to-FPGA interface na unused, a Lightweight HPS-to-FPGA na 32-bit
- U podešavanjima FPGA-to-HPS SDRAM Interface izabrati f2h\_sdram0 i zatim isključiti pritiskom na dugme '-'
- U podešavanjima Interrupts uključiti opciju Enable FPGA-to-HPS interrupts
- 11. Pod tabom Peripherial Pins izvršiti sledeće izmene
  - U podešavanjima SD/MMC Controller postaviti SDIO pin na HPS I/O Set 0 i SDIO mode na 4-bit Data
  - U podešavanjima UART Controllers postaviti UARTO pin na HPS I/O Set 0 i UART mode na No Flow Control

U ovom tabu je za potrebe nekog drugog projekta moguce uključiti ostale periferije: CAN Controller, Ethernet Media Access Controller, I2C Controller, SPI Controller, QSPI Flash Controller, NAND Flash Controller, Trace Port Intefrace Unit, GPIO za podesavanja pogledati []

- 12. Pod tabom HPS Clocks ostaviti podešavanja na podrazumevanim vrednostima
- 13. Pod tabom SDRAM podesiti:
  - SDRAM Protocol: DDR3
  - PHY Settings:
    - Clocks:
      - \* Memory clock frequency: 400.0 MHz
      - \* PLL reference clock frequency: 25.0 MHz
    - Advanced PHY Settings:
      - \* Supply Voltage: 1.5V DDR3
  - Memory Parameters:
    - Memory vendor: Other
    - Memory device speed grade: 800.0 MHz
    - Total interface width: 32
    - Number of chip select/depth expansion: 1
    - Number of clocks: 1
    - Row address width: 15
    - Column address width: 10
    - Bank-address width: 3
    - Uključiti DM pins
    - Uključiti DQS#
    - Memory Initialization Options:
      - \* Mirror Addressing: 1 per chip select: 0
      - \* Burst Length: Burst chop 4 or 8 (on the fly)
      - \* Read Burst Type: Sequential
      - \* DLL precharge power down: DLL off
      - \* Memory CAS latency setting: 11

- \* Output drive strength setting: RZQ/7
- \* ODT Rtt nominal value: RZQ/4
- \* Auto selfrefresh method: Manual
- \* Selfrefresh temperature: Normal
- \* Memory write CAS latency setting: 8
- \* Dynamic ODT (Rtt\_WR) value: RZQ/4

#### • Memory Timing:

- tIS (base): 180 ps
- tIH (base): 140 ps
- tDS (base): 30 ps
- tDH (base): 65 ps
- tDQSQ: 125 ps
- tQH: 0.38 cycles
- tDQSCK: 255 ps
- tDQSS: 0.25 cycles
- tQSH: 0.4 cycles
- tDSH: 0.2 cycles
- tDSS: 0.2 cycles
- tINIT: 500 us
- tMRD: 4 cycles
- tRAS: 35.0 ns
- tRCD: 13.75 ns
- tRP: 13.75 ns
- tREFI: 7.8 us
- tRFC: 260.0 ns
- tWR: 15.0 ns
- tWTR: 4 cycles
- tFAW: 30.0 ns
- tRRD: 7.5 ns
- tRTP: 7.5 ns

#### • Board Settings:

- Setup and Hold Derating:
  - \* Use Altera's default settings
- Channel Signal Integrity:
  - \* Use Altera's default settings

#### - Board Skews:

- \* Maximum CK delay to DIMM/device: 0.03 ns
- \* Maximum DQS delay to DIMM/device: 0.02 ns
- \* Minimum delay difference between CK and DQS: 0.06 ns
- \* Maximum delay difference between CK and DQS: 0.12 ns
- \* Maximum skew within DQS group: 0.01 ns
- \* Maximum skew between DQS groups: 0.06 ns
- \* Average delay difference between DQ and DQS: 0.05 ns

- \* Maximum skew within address and command bus: 0.02 ns
- \* Average delay difference between address and command and CK: 0.01 ns

Ovim su podešavanja HPS modula završena, izabrati Finish.

- 14. Duplim klikom u Export koloni eksportovati signale memory pod imenom hps\_0\_ddr i signale hps\_io pod imenom hps\_0\_io.
- 15. Povezati HPS sa izvorom takta kao što je prikazano na slici 5



Figure 5: Povezivanje HPS i takt signala

- 16. Iz prozora IP Catalog izabrati Processors and Peripherials > Peripherials > PIO (Parallel I/O) Intel FPGA IP. Ovim se otvara meni za podešavanje PIO IP bloka
- 17. U podešavnjima PIO IP bloka pod Basic Settings postaviti Width: 8 i Direction: Output
- 18. Preimenovati PIO blok u leds\_O. Duplim klikom u Export koloni eksportovati signale external\_connection i podesiti ime leds\_O\_external\_connection.
- 19. Povezati leds\_0 blok sa izvorom takta i resetom, zatim povezati Avalon Memory Mapped Slave pod imenom s1 sa hps\_0 interfejsom h2f\_lw\_axi\_master, kao što je prikazano na slici 6
- 20. Ponovo iz prozora IP Catalog izabrati Processors and Peripherials > Peripherials
   > PIO (Parallel I/O) Intel FPGA IP. Ovim se otvara meni za podesavanje PIO IP bloka
- 21. U podešavnjima PIO IP bloka pod Basic Settings postaviti Width: 8, Direction: Input.
- 22. U podešavanjima PIO IP bloka pod Edge capture register uključiti opciju Synchronously capture, Edge Type podesiti na ANY, i uključiti bit-clearing for edge capture register.
- 23. U podešavanjima PIO IP bloka pod Interrupt uključiti opciju Generate IRQ i izabrati IRQ Type: EDGE
- 24. Preimenovati PIO blok u keys\_O. Duplim klikom u Export koloni eksportovati signale external\_connection i podesiti ime keys\_O\_external\_connection.



Figure 6: Povezivanje leds\_0 bloka



Figure 7: Povezivanje keys\_0 bloka

- 25. Povezati keys\_0 blok sa izvorom takta i resetom, zatim povezati Avalon Memory Mapped Slave pod imenom s1 sa hps\_0 interfejsom h2f\_lw\_axi\_master i na kraju povezati irq signal na f2h\_irq0 interfejs hps\_0 bloka, kao što je prikazano na slici 7
- 26. Duplim klikom u koloni Base podesiti adresu porta s1 bloka leds\_0 i porta s1 bloka keys\_0 kao na tabeli 1.
- 27. Sačuvati Platform Designer projekat izborom File > Save i sačuvati ga pod imenom ld\_dipl\_system.qsys
- 28. Trebalo bi da se pojavi obaveštenje Save System: completed successfully. Zatim odabrati iz menija Generate > Generate HDL... U novom prozoru podesiti Create HDL design files for synthesis: VHDL i isključiti opciju Crete block symbol file (.bsf). Pokrenuti generisanje klikom na Generate. Proces bi trebalo da se završi bez

grešaka ali može imati upozorenja.

- 29. Zatvoriti Platform Designer. U prozoru Quartus-a izabrati Project > Add/Remove Files in Project... i u meniju klikom na '...' izabrati fajl ld\_dipl\_system/synthesis/ld\_dipl\_system.qip.
- 30. Izabrati File > New VHDL File i novi fajl nazvati ld\_dipl.vhd. Pod Project Navigator > Files desnim klikom na ld\_dipl.vhd izabrati Set as Top-Level Entity.
- 31. U ovom fajlu je potrebno instancirati HPS komponentu iz Platform Designer-a. Potrebno je ručno napiati ovaj fajl (primer za ovaj rad dat je u dodatku. Takođe među generisanim fajovima nalazi se deklaracija komponente ld\_dipl\_system koja može biti od pomoći (fajl ~/ld\_dipl/hw/ld\_dipl\_system/ld\_dipl\_system.cmp).
- 32. Izabrati Processing > Start > Start Analysis and Synthesis
- 33. Izabrati Tools > Tcl Scripts...

Važno: Prozor koji se otvori mora da izgleda upravo kao na slici 8 (generisani fajlovi ne smeju biti duplirani). Ukoliko su fajlovi duplirani neophodno je zatvoriti Quartus i pokrenuti ponovo. Neke verzije Quartus-a imaju ovu grešku pri detekciji tcl skripti.



Figure 8: Ispravan izgled menija

Izabrati hps\_sdram\_p0\_pin\_assignments.tcl i kliknuti Run. Ukoliko dođe do grešaka proveriti da li je izvršen prethodni korak.

34. Pokrenuti kompajliranje projekta izborom Processing > Start Compilation

# 3.2 Podešavanja okruženja i preuzimanje kompajlera

- 35. Preuzeti arhivu sa Linaro toolchain-om wget https://releases.linaro.org/components/toolchain/binaries/7.2-2017.11/arm-linux-gnueabihf/gcc-linaro-7.2.1-2017.11-x86\_64\_arm-linux-gnueabihf.tar.xz
- 36. Otpakovati preuzeti toolchain:
  tar -xf arm-linux-gnueabihf/gcc-linaro-7.2.1-2017.11-x86\_64\_arm-linux-gnueabihf.tar
  ~/ld\_dipl/sw/toolchain
- 37. Dodati putanju za toolchain u promenljivu okruženja PATH. export PATH=~/ld\_dipl/sw/toolchain/bin:\$PATH
- 38. Podesiti promenljive okruženja export ARCH=arm export CROSS\_COMPILE=arm-linux-gnueabihf-
- 39. Pokrenuti podešavanja Altera SOC EDS: cd /intelFPGA/18.0/embedded source embedded\_command\_shell.sh
- 40. Dodati putanju za Quartus alate u promenljivu okruženja PATH: export PATH=~/intelFPGA\_lite/18.0/quartus/bin/:\$PATH

# 3.3 Generisanje RBF fajla

# 3.4 Generisanje i kompajliranje Preloader-a

- 42. Pokrenuti Preloader generator komadnom: bsp-editor
- 43. Izabrati File > New HPS BSP...
- 44. U novom prozoru podesiti Preloader Settings Directory: ~/ld\_dipl/hw/hps\_isw\_handoff/ld\_dipl\_system\_hps\_0
- 45. Isključiti opciju: Use default locations i podesiti BSP target directory: ~/ld\_dipl/sw/preloader. Podešavanja bi trebalo da izgledaju kao na slici 9. Izabrati OK.
- 46. U podešavanjima pod spl.boot uključiti FAT\_SUPPORT i ostala podešavanja ostaviti na podrazumevanim vrednostima. Izabrati Generate i po završenom generisanju zatvoriti program.
- 47. Izvršiti komande za kompajliranje Preloader-a cd ~/ld\_dipl/sw/preloader make



Figure 9: Podešavanja bsp-editor-a

48. Kopirati kompajlirani Preloader: cp ~/ld\_dipl/sw/preloader/preloader-mkpimage.bin ~/ld\_dipl/sdcard/a2/

## 3.5 Generisanje Device Tree

- 49. Uporebom Alterinog alata generisati Device Tree komandom: sopc2dts -i /ld\_dipl/hw/ld\_dipl\_system.sopcinfo -o /ld\_dipl/hw/ld\_dipl\_generated.dts
  - 3.6 U-Boot
- 50. Preuzeti izvorni kod projekta U-boot komandom git clone git://git.denx.de/u-boot.git ~/ld\_dipl/sw/u-boot
- 51. Promeniti radni direktorijum ~/ld\_dipl/sw/u-boot/
- 52. Napraviti novu granu na osnovu verzije v2018.01 komandom: git checkout v2018.01 -b tmp
- 53. Konfigurisati U-Boot za DE1-SoC razvojni sistem make socfpga\_de1\_soc\_defconfig
- 54. Otvoriti meni za konfigurisanje U-Boot make menuconfig
- 55. U ovom meniju je potrebno podesiti da se pri pokretanju U-Boot pokrene naša skripta callscript. Izabrati opciju Enable a default value for bootcmd pritiskom tastera space.
- 56. U novoj opciji bootcmd value pritiskom tastera enter otvoriti novi meni i upisati tekst run callscript

Nakon ovog podešavanja bi prozor trebalo da izgleda kao na slici 10

```
U-Boot 2018.01 Configuration
                    U-Boot 2018.01 Configuration
Arrow keys navigate the menu. <Enter> selects submenus ---> (or empty
submenus ----). Highlighted letters are hotkeys. Pressing <Y>
includes, <N> excludes, <M> modularizes features.
                                                    Press <Esc><Esc> to
exit, <?> for Help, </> for Search. Legend: [*] built-in [ ]
        Boot images --->
        API
        Boot timing
        Boot media
    (2) delay in seconds before automatically booting
      ] Enable boot arguments
        Enable a default value for bootcmd
    (<mark>r</mark>un callscript) bootcmd value
        Console
        Logging
      <Select>
                  < Exit >
                               < Help >
                                                        < Load >
                                           < Save >
```

Figure 10: Konfiguracija U-Boot

- 57. Izaći iz konfiguracionog menija izborom opcije exit (strelica desno i zatim enter). Sačuvati podešavanja izborom Yes.
- 58. Otvoriti konfiguracioni .h fajl komandom: vi include/configs/socfpga\_de1\_soc.h U ovom fajlu je potrebno uneti izmene kao na slici 11

```
#define CONFIG_HW_WATCHDOG
/* Memory configurations */
#define PHYS_SDRAM_1_SIZE
                                          0x40000000
                                                           /* 1GiB */
/* Booting Linux */
#define CONFIG LOADADDR
                                  0x01000000
#define CONFIG SYS LOAD ADDR
                                 CONFIG LOADADDR
/* Ethernet on SoC (EMAC) */
/* The rest of the configuration is shared */
#include <configs/socfpga_common.h>
/* Custom env vars */
#define CONFIG_EXTRA_ENV_SETTINGS \
        "scriptfile=u-boot.scr" "\0"
"scraddr=0x2000000" "\0" \
        "callscript=fatload mmc 0:1 $scraddr $scriptfile;" \
                 "source $scraddr" "\0"
#endif /* __CONFIG_TERASIC_DE1_SOC_H__ */
```

Figure 11: Izmena U-Boot konfiguracionog fajla

- 59. Pokrenuti kompajliranje U-Boot komandom make
- 60. Kopirati kompajlirani U-Boot cp ~/ld\_dipl/sw/u-boot/u-boot.img ~/ld\_dipl/sdcard/fat32/u-boot.img

61. Otvoriti novi tekstualni fajl:

vi u-boot.script

U ovom fajlu je potrebno napisati skriptu za U-Boot. Ranije u radu je dato objašnjenje skripte, dok se ceo kod nalazi u dodatku rada.

- 62. Konvertovati napisanu skriptu u odgovarajući format komandom:
  mkimage -A arm -O linux -T script -a 0 -e 0 -n Boot\_script -d u-boot.script
  u-boot.scr
- 63. Kopirati kompajlirani U-Boot cp ~/ld\_dipl/sw/u-boot/u-boot.scr ~/ld\_dipl/sdcard/fat32/u-boot.scr

#### 3.7 Linuks kernel

- 64. Preuzeti izvorni kod Linuks kernela:
  git clone git://github.com/torvalds/linux ~/ld\_dipl/sw/linux
- 65. Promeniti radni direktorijum cd ~/ld\_dipl/sw/linux
- 66. Napraviti novu granu na osnovu verzije 4.6-rc2 komandom: git checkout 4.6-rc2 -b tmp
- 67. Konfigurisati linuks za Cyclone V arhitekturu make socfpga\_defconfig
- 68. Pokrenuti kompajliranje linuksa make zImage
- 69. Kopirati kompajlirani kernel u pripremni folder cp arch/arm/boot/zImage ~/ld\_dipl/sdcard/fat32/zImage

#### Ručno pisanje Device Tree

- 70. Početi od Device Tree fajla cp arch/arm/boot/dts/socfpga\_cyclone5\_socdk.dts arch/arm/boot/dts/socfpga\_cyclone5\_ld\_dipl.dts
- 71. Isključiti CAN podešavanjem parametra status = 'disabled'. Ovaj korak je neophodan za uspešno pokretanje sistema jer DE1-SoC nema CAN. Pogledati sliku ??
- 72. Iz Device Tree fajla generisanog Alterinim alatom izdvojiti samo najbitnije informacije i upisati u novi fajl. Pogledati sliku ??
- 73. Pokrenuti kompajliranje Device Tree fajla make socfpga\_cyclone5\_ld\_dipl.dtb
- 74. Kopirati kompajlirani Device Tree Blob u pripremni folder cp ~/ld\_dipl/sw/linux/arch/arm/boot/dts/socfpga\_cyclone5\_ld\_dipl.dtb ~/ld\_dipl/sdcard/fat32/socfpga.dtb

```
regulator-max-microvolt = <3300000>;
        };
};
/ {
        soc {
                lddipl0: lddipl00xff200000 {
                         compatible = "ld,dipl";
                         reg = <0xff200000 0x00000020>;
                         interrupts = <0 40 1>;
                };
        };
};
%can0 {
        status = "disabled";
};
&gmac1 {
        status = "okay";
        phy-mode = "rgmii";
        rxd0-skew-ps = <0>;
```

Figure 12: Izmena Device Tree fajla

## 3.8 Kompajliranje drajvera

- 75. Promeniti radni folder
  cd ~/ld\_dipl/sw/device\_driver
  U ovom folderu se nalazi izvorni fajl drajvera ld\_dipl.c i odgovarajući Makefile
- 76. Podesiti promenljivu koja ukazuje na izvorni kod linuks kernela export OUT=~/ld\_dipl/sw/linux
- 77. Pokrenuti kompajliranje komandom make
- 78. Kopirati kompajlirani drajver u pripremni direktorijum cp ~/ld\_dipl/sw/device\_driver/ld\_dipl.ko ~/ld\_dipl/sdcard/ld\_dipl.ko

# 3.9 Generisanje Root File System

- 79. Preuziti Buildroot git clone git://git.busybox.net/buildroot ~/ld\_dipl/sw/buildroot
- 80. Promeniti radni folder cd ~/ld\_dipl/sw/buildroot
- 81. Napraviti novu granu na osnovu verzije 2017.08 komandom: git checkout 2017.08 -b tmp
- 82. Pokrenuti konfiguracioni meni komandom make menuconfig
- 83. U konfoguracionim meniju izvršiti sledeće izmene
  - (a) Target options
    - (b) Target Architecture (ARM(little endian))

- (c) Target Architecture Variant (cortex-A9)
- (d) Target ABI (EABIhf)
- (e) Floating point strategy (VFPv3)
- 84. Toolchain
  - 85. Toolchain type (External toolchain)
  - 86. Toolchain (Linaro ARM 2017.11)
- 87. Toolchain origin (Pre-installed toolchain) —

```
~/ld_dipl/sw/gcc-linaro-7.2.1-2017.11-x86_64_arm-linux-gnueabihf
```

Kernel —

Linux Kernel

Pokrenuti izvršavanje komandom make

Kopirati fajl sistem u pripremni direktorijum

cp ~/ld\_dipl/sw/buildroot/output/images/rootfs.tar ~/ld\_dipl/sdcard/rootfs.tar

## 3.10 Priprema SD kartice

Ubaciti SD karticu u računar. Zaključiti pod kojim imenom se kartica pojavila u sistemskom folderu /dev (za primer je uzeto /dev/sdx )

Formatirati SD karticu komandom

sudo dd if=/dev/zero of=/dev/sdx bs=512 count=1

Particionisati SD karticu pokretanjem interaktivnog programa fdisk

sudo fdisk /dev/sdx

u kom treba uneti sledeće komande

n p 3 <default> 4095 t a2

n p 1 <default> +32M t 1 b

n p 2 <default> +512M t 2 83

W

Napraviti prazne fajl sisteme na odgovarajućim particijama

sudo mkfs.vfat /dev/sdx1

sudo mkfs.ext3 -F /dev/sdx2

Promeniti radni folder

cd ~/ld\_dipl/sdcard

Napraviti foldere na koje će se mount-ovati SD kartica

mkdir mountfat32

mkdir mountext3

Mount-ovati odgovarajuće particije

sudo mount /dev/sdx1 ~/ld\_dipl/sdcard/mountfat32

sudo mount /dev/sdx2 ~/ld\_dipl/sdcard/mountext3

Napisati Preloader na odgovarajuću particiju sudo dd if=~/ld\_dipl/sdcard/a2/preloader-mkpimage.bin of=/dev/sdx3 bs=64K seek=0

Kopirati binarne fajlove za pokretanje sistema na odgovarajuću particiju sudo cp ~/ld\_dipl/sdcard/fat32/\* ~/ld\_dipl/sdcard/mountfat32

Otpakovati fajlsistem na SD karticu taf -xf rootfs.tar mountext3

Kopirati modul drajvera na SD karticu cp ld\_dipl.ko mountext3/root/home

Uveriti se da su svi fajlovi uspešno kopirani komandom sync

Unmount-ovati sve particije umount ~/ld\_dipl/sdcard/mountfat32 umount ~/ld\_dipl/sdcard/mountext3

## 3.11 Testiranje sistema

Proverti da li su MSEL podešeni za konfigurisanje FPGA sa HSP (ima negde lepa slika)

Pruključiti DE1-SoC na napajanje

Povezati Mini USB kablom PC računar i port na DE1-SoC ploči koji je obležen sa UART

Ubaciti SD karticu u odgovarajući slot

Na PC računaru pokrenuti i podseiti minicom sudo minicom -s

U podešavanjima Serial port setup uneti izmene kao na slici 13 (može se razlikovati ime Setial device)

Figure 13: Podešavanje minicom-a

Pokrenuti DE1-SoC pritiskom crvenog tastera

Nakon pokretanja sistema potrebno je ulogovati se kao korisnik sa:q

Username:root
Password:root

Učitati modul drajvera u linuks kernel komandom:

insmod /root/home/ld\_dipl.ko

Uveriti se da su se pojavili odgovarajući sistemski fajlovi: ls /sys/bus/platform/...

Uključiti LE diode slika

Uključiti prekida slika

# 4 Zaključak

Izrada ovog rada je motivisana željom da se upoznaju konkretni alati i postupak projektovanja sa SoCFPGA sistemima. Izveštaj o radu je napisan tako da pruži kratak pregled bitnih pojmova i detaljna uputsva za reporodukciju rezultata sa osvrtom na usputne probleme. Stečeno znanje moze olakšati slušanje predmeta na master studijama, ili biti osnova za rešavanje konkretnog problema (npr. ubrzavanje algoritama kompresije i obrade slike), a autoru je olakšalo snalaženje na novom radnom mestu. Autor izražava veliku zahvalnost profesoru Lazaru Saranovcu i asistentu Strahinji Jankovicu za podršku i savete prilikom izrade diplomksog rada.

# 5 Dodatak

#### 5.1 Izvorni kod drajvera

```
/* ld_dipl.c */
#include <linux/init.h>
#include <linux/module.h>
#include <linux/io.h>
#include <linux/interrupt.h>
#include <linux/uaccess.h>
#include <linux/platform_device.h>
#include <linux/of.h>
#define DEVICE_FILE_NAME
                                "ld_dipl"
#define DRIVER_NAME
                                "lddipldrv"
#define LED_OFFSET
#define KEYS_OFFSET
#define DATA_OFFSET
                                Ω
#define INTERRUPT_MASK_OFFSET
                                8
#define EDGE_CAPTURE_OFFSET
/* prototypes */
static struct platform_driver ld_dipl_driver;
/* globals */
static int g_ld_dipl_driver_irq;
static void *g_ld_dipl_driver_base_addr;
static int g_driver_mem_base_addr;
static int g_driver_mem_size;
static ssize_t irq_mask_show(struct device_driver *driver, char *buf)
        uint32_t value;
        value = ioread32(g_ld_dipl_driver_base_addr + KEYS_OFFSET
                         + INTERRUPT_MASK_OFFSET);
        return scnprintf(buf, PAGE_SIZE, "mask_=\%u\n", value);
static ssize_t irq_mask_store(struct device_driver *driver, const char *
   buf,
                              size_t count)
{
        uint32_t value;
        sscanf(buf, "%u", &value);
        iowrite32(value,g_ld_dipl_driver_base_addr + KEYS_OFFSET
                + INTERRUPT_MASK_OFFSET);
        return count;
DRIVER_ATTR(irq_mask, (S_IWUSR | S_IRUSR), irq_mask_show, irq_mask_store
   );
static ssize_t irq_flag_show(struct device_driver *driver, char *buf)
        uint32_t value;
        value = ioread32(g_ld_dipl_driver_base_addr + KEYS_OFFSET
                        + EDGE_CAPTURE_OFFSET);
        return scnprintf(buf, PAGE_SIZE, "flags_=\%u\n", value);
```

```
}
static ssize_t irq_flag_store(struct device_driver *driver, const char *
   buf,
                              size_t count)
{
       uint32_t value;
        sscanf(buf, "%u", &value);
        iowrite32(value,g_ld_dipl_driver_base_addr + KEYS_OFFSET
                + EDGE_CAPTURE_OFFSET);
       return count;
}
DRIVER_ATTR(irq_flag, (S_IWUSR | S_IRUSR), irq_flag_show, irq_flag_store
static ssize_t keys_show(struct device_driver *driver, char *buf)
       uint32_t value;
       value = ioread32(g_ld_dipl_driver_base_addr + KEYS_OFFSET
                        + DATA_OFFSET);
        return scnprintf(buf, PAGE_SIZE, "keys_=_\%u\n", value);
}
DRIVER_ATTR(keys, (S_IRUSR), keys_show, NULL);
static ssize_t leds_store(struct device_driver *driver, const char *buf,
                              size_t count)
{
       uint32_t value;
       sscanf(buf, "%u", &value);
        iowrite32(value,g_ld_dipl_driver_base_addr + LED_OFFSET +
           DATA_OFFSET);
       return count;
}
DRIVER_ATTR(leds, (S_IWUSR), NULL, leds_store);
static irqreturn_t ld_dipl_isr(int irq, void *data)
       uint32_t value;
        value = ioread32(g_ld_dipl_driver_base_addr + KEYS_OFFSET
                        + EDGE_CAPTURE_OFFSET);
       pr_info("irqureceived:u%u\n", value);
        iowrite32(value, g_ld_dipl_driver_base_addr + KEYS_OFFSET
                + EDGE_CAPTURE_OFFSET);
       return IRQ_HANDLED;
}
static struct of_device_id ld_dipl_of_match[] = {
        .compatible = "ld,dipl"
        };
```

```
MODULE_DEVICE_TABLE(of, ld_dipl_of_match);
static int ld_dipl_probe(struct platform_device *pdev)
        int ret;
        struct resource *res;
        struct resource *driver_mem_region;
        pr_info("probe_enter\n");
        ret = -EINVAL;
        /* get memory resource */
        res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
        if (res == NULL) {
                pr_err("IORESOURCE_MEM, __O_does_not_exist\n");
                return ret;
        }
        g_driver_mem_base_addr = res->start;
        g_driver_mem_size = resource_size(res);
        /* reserve our memory region */
        driver_mem_region = request_mem_region(g_driver_mem_base_addr,
                                                      g_driver_mem_size,
                                                         demo_driver_hw_region
                                                         ");
        if (driver_mem_region == NULL) {
                pr_err("request_mem_region ifailed \n");
                return ret;
        }
        /* ioremap memory region */
        g_ld_dipl_driver_base_addr = ioremap(g_driver_mem_base_addr,
           g_driver_mem_size);
        if (g_ld_dipl_driver_base_addr == NULL) {
                pr_err("ioremap | failed \n");
                goto bad_exit_release_mem_region;
        }
        /* get interrupt resource */
        g_ld_dipl_driver_irq = platform_get_irq(pdev, 0);
        if (g_ld_dipl_driver_irq < 0) {</pre>
                pr_err("invalid_IRQ\n");
                goto bad_exit_iounmap;
        }
        pr_info("interrutp_is:_\d\n", g_ld_dipl_driver_irq);
        /* register interrupt handler */
        ret = request_irq(g_ld_dipl_driver_irq,
                               ld_dipl_isr,
                               ld_dipl_driver.driver.name,
                               &ld_dipl_driver);
        if (ret < 0) {</pre>
                pr_err("unable_to_request_IRQ\n");
                goto bad_exit_iounmap;
        }
```

```
/* create the sysfs entries */
        ret = driver_create_file(&ld_dipl_driver.driver,
                                      &driver_attr_irq_mask);
        if (ret != 0) {
                pr\_err("failed_{\sqcup}to_{\sqcup}create_{\sqcup}irq\_mask_{\sqcup}sysfs_{\sqcup}entry");
                 goto bad_exit_remove_irq_mask;
        }
        ret = driver_create_file(&ld_dipl_driver.driver,
                                      &driver_attr_keys);
        if (ret != 0) {
                pr_err("failed_uto_ucreate_keys_sysfs_entry");
                 goto bad_exit_remove_keys;
        }
        ret = driver_create_file(&ld_dipl_driver.driver,
                                      &driver_attr_leds);
        if (ret != 0) {
                pr_err("failed_to_create_leds_sysfs_entry");
                 goto bad_exit_remove_leds;
        }
        ret = driver_create_file(&ld_dipl_driver.driver,
                                      &driver_attr_irq_flag);
        if (ret != 0) {
                 pr_err("failed_to_create_irq_flag_sysfs_entry");
                 goto bad_exit_remove_irq_flag;
        pr_info("probe_exit_success\n");
        return 0;
bad_exit_remove_irq_flag:
        driver_remove_file(&ld_dipl_driver.driver,
                            &driver_attr_irq_flag);
bad_exit_remove_leds:
        driver_remove_file(&ld_dipl_driver.driver,
                            &driver_attr_leds);
bad_exit_remove_keys:
        driver_remove_file(&ld_dipl_driver.driver,
                            &driver_attr_keys);
bad_exit_remove_irq_mask:
        driver_remove_file(&ld_dipl_driver.driver,
                            &driver_attr_irq_mask);
bad_exit_freeirq:
        free_irq(g_ld_dipl_driver_irq, &ld_dipl_driver);
bad_exit_iounmap:
        iounmap(g_ld_dipl_driver_base_addr);
bad_exit_release_mem_region:
        release_mem_region(g_driver_mem_base_addr, g_driver_mem_size);
        pr_info("probe_exit_fail\n");
        return ret;
}
static int ld_dipl_remove(struct platform_device *pdev)
        pr_info("remove_enter\n");
        driver_remove_file(&ld_dipl_driver.driver,
                            &driver_attr_irq_flag);
        driver_remove_file(&ld_dipl_driver.driver,
```

```
&driver_attr_leds);
        driver_remove_file(&ld_dipl_driver.driver,
                           &driver_attr_keys);
        driver_remove_file(&ld_dipl_driver.driver,
                           &driver_attr_irq_mask);
        driver_remove_file(&ld_dipl_driver.driver,
                           &driver_attr_irq_flag);
        driver_remove_file(&ld_dipl_driver.driver,
                           &driver_attr_leds);
        driver_remove_file(&ld_dipl_driver.driver,
                           &driver_attr_keys);
        driver_remove_file(&ld_dipl_driver.driver,
                           &driver_attr_irq_mask);
        free_irq(g_ld_dipl_driver_irq, &ld_dipl_driver);
        iounmap(g_ld_dipl_driver_base_addr);
        release_mem_region(g_driver_mem_base_addr, g_driver_mem_size);
        pr_info("remove_exit\n");
        return 0;
}
static struct platform_driver ld_dipl_driver = {
        .probe = ld_dipl_probe,
        .remove = ld_dipl_remove,
        .driver = {
                .name = DRIVER_NAME,
                .of_match_table = ld_dipl_of_match,
        },
}:
static int __init ld_dipl_init(void)
        int ret;
        pr_info("init_enter\n");
        ret = platform_driver_register(&ld_dipl_driver);
        if (ret != 0)
                pr_err("platform_driver_register_returned_%d\n", ret);
        pr_info("init_exit\n");
        return ret;
}
static void __exit ld_dipl_exit(void)
{
        pr_info("ld_dipl_exit_enter\n");
        platform_driver_unregister(&ld_dipl_driver);
        pr_info("ld_dipl_exit\n");
}
module_init(ld_dipl_init);
module_exit(ld_dipl_exit);
MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("ETFuDiplomauThesisuFPGAuDriveruanduDevice");
MODULE_AUTHOR("Dejan_Lukic");
```

# 5.2 Skripta za U-Boot

```
echo --- Reseting env variables... ---
# reset environment variables to default
env default -a
echo --- Setting env variables... ---
# Set kernel image
setenv bootimage zImage;
# adress to wich the device tree will be loaded
setenv fdtaddr 0x00000100
# set device tree image
setenv fdtimage socfpga.dtb;
#set kernel boot arguments, boot kernel
setenv mmcboot 'setenvubootargsumem=1024Muconsole=ttyS0,115200uroot=${
  #load linux kernel image and device tree to memory
setenv mmcload 'mmcurescan;u${mmcloadcmd}ummcu0:${mmcloadpart}u${
   }<sub>□</sub>${fdtimage}'
#command to be executed to read from sdcard
seteny mmcload fatload
# sdcard fat32 partition number
setenv mmcloadpart 1
# sdcard ext3 identifier
setnev mmcroot /dev/mmcblk0p2
#standard io
setenv stderr serial
setenv stdin serial
setenv stdout serial
#save environment to sdcard
saveenv
echo --- Programming FPGA...---
#load rbf from fat partition to memory
fatload mmc 0:1 ${fpgadata} socfpga.rbf;
# progran fpga
fpga load 0 ${fpgadata} #{filesize}
# enable h2f, f2h, lw_h2f
bridge enable;
echo --- Booring Linux...---
#load kernel and device tree to memory
run mmcload;
```

# set bootargs, boot kernel
run mmcboot;