

## FELADATKIÍRÁS

A feladatkiírást a tanszéki adminisztrációban lehet átvenni, és a leadott munkába eredeti, tanszéki pecséttel ellátott és a tanszékvezető által aláírt lapot kell belefűzni (ezen oldal *helyett*, ez az oldal csak útmutatás). Az elektronikusan feltöltött dolgozatban már nem kell beleszerkeszteni ezt a feladatkiírást.



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

Villamosmérnöki és Informatikai Kar

Méréstechnika és Információs Rendszerek Tanszék

# A NES videojáték konzol FPGA alapú megvalósítása

DIPLOMATERV

*Készítette*  
Varga Dominik

*Konzulens*  
Raikovich Tamás

2023. november 29.

# Tartalomjegyzék

## Kivonat

### Abstract

|                                                                                       |           |
|---------------------------------------------------------------------------------------|-----------|
| <b>1. Bevezetés</b>                                                                   | <b>1</b>  |
| <b>2. Felhasznált eszközök</b>                                                        | <b>2</b>  |
| 2.1. Altium Designer . . . . .                                                        | 2         |
| 2.2. Xilinx ISE . . . . .                                                             | 2         |
| 2.3. Logsys GUI . . . . .                                                             | 2         |
| <b>3. Nintendo Entertainment System ismertetése</b>                                   | <b>3</b>  |
| 3.1. NES hardver főbb komponensei . . . . .                                           | 4         |
| 3.1.1. Komponensek kapcsolata - PPU és CPU adatbusz . . . . .                         | 6         |
| 3.2. Képalkotás - Picture process unit . . . . .                                      | 6         |
| 3.2.1. PPU által generált kimeneti jel . . . . .                                      | 7         |
| 3.2.2. Pattern táblák és paletták . . . . .                                           | 8         |
| 3.2.3. A név táblák (Name Tables) és tulajdonság táblák (Attribute Tables)            | 10        |
| 3.2.4. Objektum Attribútum Memória (OAM) . . . . .                                    | 11        |
| 3.2.5. PPU hardveres hibái (bug-ok) . . . . .                                         | 11        |
| 3.3. 2A03 a NES fő vezérlő egysége . . . . .                                          | 12        |
| 3.3.1. MOS 6502 - központi feldolgozó egység (Central Processing Unit, CPU) . . . . . | 12        |
| 3.3.2. Audió feldolgozó egység (Audio Process Unit, APU) . . . . .                    | 15        |
| 3.3.3. Direct Memory Access - OAM DMA . . . . .                                       | 16        |
| 3.4. NES kontroller . . . . .                                                         | 17        |
| <b>4. NES FPGA alapú újra gondolása</b>                                               | <b>18</b> |
| 4.1. Képalkotás . . . . .                                                             | 18        |
| 4.2. Audio . . . . .                                                                  | 19        |
| 4.3. Játékok tárolása . . . . .                                                       | 19        |
| 4.4. Kompakt hordozható méret . . . . .                                               | 20        |
| <b>5. FPGA NES kártya ismertetése</b>                                                 | <b>21</b> |
| 5.1. Tápellátás . . . . .                                                             | 23        |
| 5.2. Órajel források . . . . .                                                        | 23        |
| 5.3. Memória - SRAM . . . . .                                                         | 23        |
| 5.4. Digital Analog Converter és erősítő . . . . .                                    | 23        |
| 5.5. HDMI és I2C szint illesztő . . . . .                                             | 25        |
| 5.6. A kártya bemenetei . . . . .                                                     | 25        |
| 5.7. MicroSD kártya . . . . .                                                         | 26        |

|                                                                 |           |
|-----------------------------------------------------------------|-----------|
| 5.8. FPGA konfigurációs módok . . . . .                         | 26        |
| 5.9. Soros Flash memória . . . . .                              | 27        |
| 5.10. LOGSYS fejlesztői port . . . . .                          | 27        |
| 5.11. Nyomtatott áramköri terv . . . . .                        | 27        |
| 5.11.1. Réteg beállítások . . . . .                             | 27        |
| 5.11.2. Komponensek elhelyezése . . . . .                       | 28        |
| 5.11.3. HDMI adatvonalainak bekötése . . . . .                  | 30        |
| 5.11.4. FPGA táp vonalak kialakítása . . . . .                  | 31        |
| 5.11.5. FPGA NES 3D terve . . . . .                             | 32        |
| <b>6. FPGA tervezés</b>                                         | <b>33</b> |
| 6.1. Rendszer blokkvázlat bemutatása . . . . .                  | 33        |
| 6.1.1. Adatbuszok implementálása . . . . .                      | 34        |
| 6.1.2. Működési órajel választása . . . . .                     | 35        |
| 6.2. Picture Process Unit . . . . .                             | 35        |
| 6.2.1. Eredeti rendrelés menete . . . . .                       | 35        |
| 6.2.2. VGA rendelés . . . . .                                   | 37        |
| 6.2.3. Háttér renderelési állapot gép . . . . .                 | 37        |
| 6.2.4. Sprite rendering állapot gép . . . . .                   | 37        |
| 6.2.5. CPU által elérhető regiszterek és CPU adatbusz . . . . . | 37        |
| 6.2.6. PPU adatbusz és memória elérése . . . . .                | 37        |
| 6.3. NES memória felépítése FPGA-ban . . . . .                  | 37        |
| 6.4. DMA . . . . .                                              | 37        |
| 6.5. 6502 processzor működése . . . . .                         | 37        |
| 6.6. NES kontrollerek kezelése . . . . .                        | 37        |
| <b>7. A NES tesztelése</b>                                      | <b>38</b> |
| 7.1. Rendszer szimulációk . . . . .                             | 38        |
| 7.2. Hardveres tesztek . . . . .                                | 38        |
| 7.2.1. Donkey Kong . . . . .                                    | 38        |
| 7.2.2. Super Mario Bros. . . . . .                              | 38        |
| 7.3. A tesztek eredményeinek kiértékelése . . . . .             | 38        |
| <b>8. Összefoglalás, jövőbeli tervezek</b>                      | <b>39</b> |
| <b>Köszönhetetlenül</b>                                         | <b>40</b> |
| <b>Ábrák jegyzéke</b>                                           | <b>41</b> |
| <b>Táblázatok jegyzéke</b>                                      | <b>42</b> |
| <b>Irodalomjegyzék</b>                                          | <b>42</b> |
| <b>Függelék</b>                                                 | <b>44</b> |
| F.1. NES kártya alkatrész elhelyezési terve . . . . .           | 44        |
| F.1.1. Top . . . . .                                            | 44        |
| F.1.2. Bottom . . . . .                                         | 45        |
| F.2. Nyomtatott áramköri terv (2D transparent) . . . . .        | 46        |
| F.3. FPGA NES kártya kapcsolási rajza . . . . .                 | 47        |
| F.3.1. Tápegység . . . . .                                      | 47        |
| F.3.2. HDMI és MicroSD kártya csatlakozó . . . . .              | 48        |
| F.3.3. DAC, erősítő és kontroller áramkörök . . . . .           | 49        |

|        |                   |    |
|--------|-------------------|----|
| F.3.4. | SRAM és SPI-Flash | 50 |
| F.3.5. | FPGA OSC és JTAG  | 51 |
| F.3.6. | FPGA IO bankok    | 52 |

## HALLGATÓI NYILATKOZAT

Alulírott *Varga Dominik*, szigorló hallgató kijelentem, hogy ezt a diplomatervet meg nem engedett segítség nélkül, saját magam készítettem, csak a megadott forrásokat (szakirodalom, eszközök stb.) használtam fel. minden olyan részt, melyet szó szerint, vagy azonos értelemben, de átfogalmazva más forrásból átvettem, egyértelműen, a forrás megadásával megjelöltem.

Hozzájárulok, hogy a jelen munkám alapadatait (szerző(k), cím, angol és magyar nyelvű tartalmi kivonat, készítés éve, konzulens(ek) neve) a BME VIK nyilvánosan hozzáférhető elektronikus formában, a munka teljes szövegét pedig az egyetem belső hálózatán keresztül (vagy autentikált felhasználók számára) közzétegye. Kijelentem, hogy a benyújtott munka és annak elektronikus verziója megegyezik. Dékáni engedéllyel titkosított diplomatervek esetén a dolgozat szövege csak 3 év eltelte után válik hozzáférhetővé.

Budapest, 2023. november 29.

---

*Varga Dominik*  
hallgató

# Kivonat

Jelen dokumentum egy diplomaterv sablon, amely formai keretet ad a BME Villamosmérnöki és Informatikai Karán végző hallgatók által elkészítendő szakdolgozatnak és diplomaternevnek. A sablon használata opcionális. Ez a sablon L<sup>A</sup>T<sub>E</sub>X alapú, a *TeXLive* T<sub>E</sub>X-implementációval és a PDF-L<sup>A</sup>T<sub>E</sub>X fordítóval működőképes.

# Abstract

This document is a L<sup>A</sup>T<sub>E</sub>X-based skeleton for BSc/MSc theses of students at the Electrical Engineering and Informatics Faculty, Budapest University of Technology and Economics. The usage of this skeleton is optional. It has been tested with the *TeXLive* T<sub>E</sub>X implementation, and it requires the PDF-L<sup>A</sup>T<sub>E</sub>X compiler.

## 1. fejezet

# Bevezetés

Az 1983-ban megjelent Nintendo Entertainment System (NES) 8 bites videojáték konzol a maga korában igen népszerű volt. A hardverének kialakítása több későbbi, modernebb videojáték konzolra volt hatással, valamint számos kiemelkedő játékprogram erre a konzolra készült el először.

A NES viszonylag egyszerű, jól átgondolt hardvere lehetővé teszi annak az olcsóbb, kevesebb erőforrással rendelkező FPGA eszközökkel történő megvalósítását. Egy ilyen megvalósításnak több előnye is van, például ki lehet használni a modern megjelenítő interfések (VGA, DVI, HDMI, stb.), valamint az eredeti játékkazetta helyett alkalmazni lehet modern adattároló eszközöket is (SD kártya). Természetesen ezen „továbbfejlesztések” mellett az egyedi változat képes futtatni az eredeti játékprogramokat.

A feladat célja egy egyedi, FPGA alapú NES megvalósítás hardver és szoftver komponenseinek elkészítése.

## **2. fejezet**

# **Felhasznált eszközök**

### **2.1. Altium Designer**

### **2.2. Xilinx ISE**

A Xilinx ISE (Integrated Softver Environment) a Xilinx Inc. által kifejlesztett, széles körben használt szoftvercsomag volt. Átfogó eszközökészletet biztosított a digitális logikai áramkörök tervezéséhez, teszteléséhez és megvalósításához a Xilinx Field-Programmable Gate Array (FPGA) és Complex Programmable Logic Device (CPLD) eszközökkel.

A Xilinx ISE teljes körű megoldást kínált a digitális tervezéshez, beleértve a HDL (Hardware Description Language) tervezést, a szimulációt, a szintézist, az implementációt és az eszközprogramozást. Támogatta a különböző tervezési beviteli módszereket, beleértve a Xilinx Schematic Editor segítségével történő sematikus rögzítést és a HDL-alapú tervezést olyan nyelvekkel, mint a VHDL (VHSIC Hardware Description Language) és a Verilog.

### **2.3. Logsys GUI**

### 3. fejezet

## Nintendo Entertainment System ismertetése

A Nintendo Entertainment System (NES) egy otthoni videojáték-konzol, amelyet a Nintendo 1983-ban Japánban (Family Computer, röviden FamiCom néven) és 1985-ben Észak-Amerikában, Európában és Ausztráliában adott ki. Ez minden idők egyik legikonikusabb és legnagyobb hatású videojáték-konzolja.

A NES döntő szerepet játszott a videojáték-ipar újjáélesztésében az 1983-as észak-amerikai videojáték-válság után. Számos klasszikus és kedvelt játékot mutatott be, amelyeket a játékosok még ma is nagyra tartanak. A konzol sikere az erős játéktárnak, a felhasználóbarát kialakításnak és az innovatív marketingstratégiáknak köszönhető.



**3.1. ábra.** Nintendo Entertainment System

Az otthoni konzol 8 bites processzorral rendelkezett, és a játékok tárolására elsősorban kazettákat használt. Jellegzetes, téglalap alakú kialakítása volt, a játékkazetták behelyezésére szolgáló elülső betöltő mechanizmussal. A konzolhoz egy pár kontroller is tartozott, és bevezette a ma már ikonikus NES kontroller kialakítását, amely irányjelzővel, start- és választógombokkal, valamint az A és B gombokkal rendelkezett.

A konzolra megjelent legnépszerűbb és legnagyobb hatású játékai közé tartozik a Super Mario Bros., a The Legend of Zelda, a Metroid, a Mega Man, a Castlevania és még sok más játék. Ezek a játékok megalapoztak számos sikeres franchise-t, amelyek ma is virágognak.

A játékkonzol hardverének megismerésére, elsősorban a hivatalos wiki oldalt használtam [3]. Az itt olvasható tartalmakat az évek során nagy részt az eredeti hardver visszafejtésével (reverse engineering) tárta fel, mivel a játékkonzol pontos adatlapjai, illetve időzítés és működési diagramjai nem lettek publikusak (a Nintendo tulajdonban vannak). Az oldalon szereplő adatokat a NESDev online közösség tartja karban, ezáltal az oldal pontos és helyes adatokat tartalmaz (ezeket a közösség rendszeresen felül vizsgálja). A kutatáshoz sok segítséget nyújtott Nathon Altice által írt könyv [1]. Ez rengeteg fundamentális információt nyújt a NES hardverén belül található chipekről. A NES hardverének áttekintéséhez és a chipek általános megismeréséhez (PPU, CPU) nagy segítséget nyújtott még a NESHacker youtube csatorna [6]. Amely páratlanul pontos információkkal és képi adatokkal szolgált az eszköz működéséről.

### 3.1. NES hardver főbb komponensei

A Nintendo Entertainment System hardverének áttekintéséhez célszerű az eredeti konzol alaplapjának tüzetes vizsgálata. Ezt a 3.2 képen láthatjuk, és ez alapján a komponensek öt fő csoportba sorolhatjuk.



**3.2. ábra.** Nintendo Entertainment System NTSC alaplap [5]

- Sötétkék, processzor:* A NES processzora és ennek kiegészítő áramkörei, komponensei. Ide tartozik természetesen az RP2A03G chip, ez a fő vezérlő egyégy két fő elemet tartalmaz : egy átalakított 6502-es processzort, illetve az APU co-processzort amely a hang generálásért felelős. Ezen a kék területen belül még megtalálható a WRAM (cpu alatt) amely egy két kilobájt méretű rendszer memória (system memory) ként volt használva, illetve az 74LS139-is (cpu-tól balra fent) amely a chip kiválasztó

(chip select) jelek elő állításáért felelős hardver. Itt kapott még helyet a keret tetején látható kék oszcillátor kristály is és ennek segéd áram köre. Ez a komponens látta el az egész alaplapot órajellel.

- *Narancssárga, videó generálás:* Ebbe a kategóriába három komponens tartozik. Első az RP2C02G-O chip amely a videó generálás fő vezérlő egysége volt, ennek neve Picture Process Unit (PPU). A PPU alatt pedig a két memória chip látható, az alsó raktározta el a képgeneráláshoz szükséges adatokat (VRAM), a fölötté lévő egy adatpufferként funkcionált.
- *Citromsárga, CIC chip:* Itt található a CIC chip amely azért volt felelős, hogy nem licencelt játékokat ne lehessen a NES-el játszani. Illetve itt még egy invertáló áramkör kapott helyet (ezt nagyjából minden alaplapi komponens használta ha bit invertálásra volt szüksége).
- *Lila, kompozit kimenet:* Ezen a területen láthatók a kompozit jel elő állításáért felelős áramköri elemek és a táp bemenet szűréséért felelős kapacitások.
- *Világoskék, kontrollerek:* A fenti csatlakozó az egyes játékos kontroller portja a jobb oldali pedig a második játékosé. Itt még látható két dedikált invertáló komponens is, amelyek a kontrollerekből érkező adatok negálásával foglalkoztak.

A fenti elemek kívül az alaplapon még található középen egy bővítő csatlakozó is, illetve a kártya aljára lehet közvetlenül csatlakoztatni a játék kazettákat is. A játék kazetták multifunkcionális nyákok voltak és általában a 3.3 ábrán látható módon néztek ki.



**3.3. ábra.** Nintendo Entertainment System játék kazetta [4]

A játék kazetták fő alkotó elemei:

- *Program ROM, és Mapper-ek:* A játék szoftvert tartalmazó, kezdetben 16 kilobájt - 32 kbyte méretű ROM. Viszont a 32 kilobájtnál nagyobb játékok esetén a ROM

egy extra segéd chippel (Mapper) egészült ki, amelyen keresztül a NES megtudta címzni a nagyobb ROM területet.

- *Karakter ROM:* Általában 8 kilobájt méretű ROM. A megjelenítéshez szükséges csempe elemeket tárolja (3.6).
- *CIC chip:* Ez a komponens licencelte a játék szoftverét a NES konzol felé.

### 3.1.1. Komponensek kapcsolata - PPU és CPU adatbusz

A különböző komponensek mélyebb bemutatása előtt a NES hardverének adat kapcsolatát is át kell tekintenünk. A NES úgynevezett memory-mapped I/O architektúrával rendelkezik. Ez egy olyan technika amely a rendszer teljes memória területét feldarabolja nagyobb egységekre, és ezeket hardveres komponensekhez rendel. Ezek alapján az alaplapon található WRAM, a játék kártyán található Program ROM, illetve a PPU hét vezérlő regisztere és az APU vezérlőregisztere, mind a CPU 16 bites címterületén belül található és a CPU adatbuszon keresztül írható és olvasható. A NES-en belül található még egy PPU adatbusz is amely a VRAM-ot és a játék kazettán található karakter ROM-ot foglalja a PPU cím tartományba (3.1). Ezeket az adatkapcsolatokról a 3.4 sematikus ábra foglalja össze.



**3.4. ábra.** Komponensek közti kapcsolat (fent játék kazetta) [5]

## 3.2. Képalkotás - Picture process unit

A következőkben azt fogom bemutatni, hogy a NES hogyan tárol, dolgoz fel és jelenít meg sprite grafikákat. A Sprite egy 8x8-as pixel csempét jelent, ez a NES képalkotásának alapjállítre.

A NES főkomponensei közül a Picture Process Unit (későbbiekben PPU) felelős a konzol 8-bit-es grafikájának elő állításáért. A PPU egy a Nintendo által kifejlesztett speciális chip amely a processzor mellett működik, mint egy társprocesszor (coprocessor), hasonlóan a napjainkban elterjedt videókártya-processzor pároshoz.

A CPU-tól eltérően a PPU egy előre meghatározott grafikus műveleti parancs sorozatot hajt végre ciklikusan, nem lehet közvetlenül programozni. Saját memóriával rendelkezik amelyet a CPU képes módosítani, hogy ezzel megváltoztassa a grafika generálását. Ez a memóriaterület a következőképpen négy részre oszlik:

- *Pattern táblák:* Az első szekció tartalmazza a pattern táblákat, amelyek a nyers sprite-kép adatokat tartalmazzák az adott játékhoz. Két pattern tábla van a bal

oldali és a jobb oldali tábla amelyek mindegyike 64 kilobyte-nyi memória. Együttesen pedig 256 darab 8 x 8 pixeles csempét tárolnak. A memória ezen része általában közvetlenül a játék kazetta karakter ROM vagy RAM chipjére van leképezve.

- *Névtáblák (Nametables)*: A következő rész a PPU névtábláit tartalmazza, amelyek a háttérgrafikák kialakítására szolgálnak a játékhoz. Ezek 32x30-as rászterben vannak felépítve, a rászter minden egyes eleme egy 8x8 pixeles területet reprezentál a képernyőn. A cellák egyetlen byte-ot tartalmaznak, amely egy csempét címez meg a Pattern táblákban.
- *Paletták (Palettes)*: A harmadik rész az aktív szín paletták tárolására szolgál a játékhoz. A PPU képes több mint 50 különböző szín előállítására, de nem tudja az összes színt egyszerre használni egyidejűleg, ehelyett ezt a memória területet arra használják, hogy meghatározzunk nyolc aktív palettát amelyek egyenként négy színt tartalmaznak. Ebből a nyolc palettából, választhatunk színt a pixel-ek megjelenítése során.
- *Objektum Attribútum Memória (későbbiekben OAM)*: A PPU memóriának ez a része vezérli a játék előtérben lévő grafikájának megjelenítését. Ezek olyan dolgok, mint például Mario, Link, az ellenségek és az olyan effektek, mint a tűzgolyók és robbanások. Alapvetően bármi, ami a háttér grafika felett vagy néha alatta jelenne meg.

Tehát, mindez összegezve, úgy tekinthetjük a PPU-ra, mintha ez a négy jól elkülöníthető memória terület irányítaná ezt a segéd processzort. A Pattern táblák határozzák meg a nyers képadatokat. A névtáblák határozzák meg a háttér generálását. A színpaletták határozzák meg a használandó színeket és az OAM vezérli az előtérbe vagy háttérbe kerülő mozgó sprite-okat. Ezen felül a PPU további funkciókkal is rendelkezik, ezeket nyolc különböző regiszter írásával és olvasásával érhetjük el. Ezekről a regiszterekről az implementálás során a 6.2 fejezetben még olvashatunk.

### 3.2.1. PPU által generált kimeneti jel

Ebben a fejezetben bemutatom a PPU által generált jelet és ennek felhasználást a régi típusú CRT TV-kben. A CRT televízió (az eredeti TV) a modern lapos képernyők előfutára volt, alapvetően két fő komponensből épültek fel egy fluoreszkáló képernyőből és egy katódsugárcsőből.

A CRT működése röviden: a katódsugárcső egy pisztolyként funkcionál, amely elektro-nokat lő ki a képernyőre és amikor elég elektron találja el a képernyő egy bizonyos területét az világítani kezd. A televíziók kétféle típusban léteztek, fekete-fehérben vagy színesben. A fekete-fehér esetben egy elektronágyú szabályozta a képernyő pixeljeinek monokróm fényerejét, a színes esetben három különálló elektronágyú szabályozta a vörös, kék és zöld komponensek arányát, ezzel megalkotva a színes képet. A színes TV-k esetében is a három elektron sugár együtt mozgott végig a képernyőn, ezért a könnyebb megértés érdekében érdemes egy elektron sugárként gondolni ezekre.

A televízió működése során a bal felső sarokból kezdve úgy irányítja az elektron sugarat, hogy a teljes képernyőn végigfusson sorról sorra, amíg el nem éri a jobb alsó sarkát a képernyőnek. Ha egy sor végére érünk akkor az elektron sugarat vissza pozicionáljuk a sor elejére, ezt az időt horizontális szinkronizációjának nevezzük (horizontal blanking). Ha végigértünk egy képkockán a TV a katódsugárcsövet újra a felső sor bal oldalára állítja, ezt vertikális szinkronizációjának (vertical blanking) hívják. Ez képalkotási ciklus a TV működése közben rögzített időközönként ismétlődik, általában másodpercenként hatvan képkocka körül.



**3.5. ábra.** Katódsugárcsöves TV-k működése

Miközben a elektronágú mozog, a TV egy belső jel segítségével tudja szabályozni az elektronok kibocsátásának mértékét. Ez a jel megváltoztatja a szín fényerejét egy adott pozícióban (pixel-en), a pisztoly gyors mozgásának következtében, az eredmény egy folytonos animált kép képernyőn. Ezt a jelet kompozit jelnek nevezzük és általában egy rf antennáról vagy egy kábel dobozból származott, de a NES esetében ezt a jelet a PPU állítja elő.

A NES két típusú kompozit jelet tudott előállítani, attól függően, hogy a világ melyik területére gyártották. Ez azért következett be mert a CRT TV-knek két standard típusa terjedt el világszerte az NTSC és a PAL. Az NTSC-t elsősorban az Egyesült Államokban és Japánban használták, és 60 képkocka/másodperces sebességgel jelenítette meg kompozit jelet, ezek a TV-k összesen 525 képsorral rendelkeztek. A PAL-t elsősorban Európában, Afrikában és Dél-Amerikában használták és 50 képkocka/másodperc sebességgel futott, összesen 625 képsort jelenített meg.

A NES játékokat sosem programozták régió specifikusak, az egyetlen dolog ami változott területenként az a NES PPU-jának hardvere. Így a világ különböző területein ugyanaz a játék gyorsabban illetve lassabban futott, ez akár 17%-os különbséget is jelenthetett. A NES emulálás szempontjából az NTSC készülékeket veszem alapul mivel ezeken gépek működését tárták fel részletesebben reverse engineering-gel.

### 3.2.2. Pattern táblák és paletták

A sprite képek, amelyek a PPU Pattern tábla memóriájában helyezkednek el, képezik az alapját minden grafikai megjelenítésnek. Ezek a 8 x 8 pixeles csempék alkotják a bonyolult háttereket, mozgó objektumokat és speciális effekteket. Alapvetően a sprite-ok tárolása hasonló módon történik mint egy modern számítógépek által használt kép esetében, mint például png. Mindkét tárolási formátumot kétdimenziós szám rácsként tudjuk elközelni, ahol minden egyes cellához tartozó érték egy pixelhez tartozó színt reprezentál, csak amíg a png több millió színt támogat addig a NES egy pixel-e csupán négy különböző színű lehet. Gyakorlatilag egy ilyen csempe nem is tárol szín adatokat, a pixel-eket reprezentáló számértékek referenciák egy éppen aktív szín paletta színére.



**3.6. ábra.** Super Mario Bros. Pattern táblái

A színpaletták memória területének írásával, nyolc különböző palettát állíthatunk be a grafikus megjelenítéshez, négyet a háttér megjelenítéséhez és a maradék négyet az előtér rendereléséhez. minden egyes paletta négy színt tárol, az első szín minden egyes palettán egy átlátszósági szín, ez azt jelenti, hogy ha a pixel szín értéke nullás indexel rendelkezik akkor az megjelenítés során minden esetben átlátszó lesz (függetlenül a palettába írt szín értéktől).

|    |    |    |    |    |    |    |    |
|----|----|----|----|----|----|----|----|
| 00 | 00 | 00 | 00 | 00 | 00 | 01 | 01 |
| 00 | 00 | 00 | 00 | 00 | 01 | 01 | 01 |
| 00 | 00 | 00 | 00 | 01 | 01 | 01 | 01 |
| 00 | 00 | 00 | 01 | 01 | 01 | 01 | 01 |
| 00 | 00 | 01 | 11 | 11 | 01 | 01 | 01 |
| 00 | 01 | 01 | 01 | 10 | 11 | 01 | 01 |
| 00 | 01 | 01 | 01 | 10 | 11 | 11 | 11 |
| 01 | 01 | 01 | 01 | 10 | 11 | 10 | 01 |

**3.7. ábra.** Gumba (Mario egyik ellenfele) bal felső csempe elem szín adatai [?]

A fentiek alapján, tehát egy pixel 0-3-ig vehet fel értéket, tehát 2 biten vagyunk képesek eltárolni az értékét. Egy 8 x 8 as csempe pedig 64 pixelt tartalmaz, ezért összesen 16 byte helyet foglal. Mivel a NES processzora egy 6502-es (8 bit-es) modell volt, ennek okán a legkisebb memóriaegység amit képes volt megcímzni az egy byte volt, így a csempe adatok ésszerű tárolásához egyedi megoldásra volt szükség.

Mivel a NES nem képes direkt a 2 bit-es számértékekkel dolgozni, ezért ezeket fel daraboljuk először logikailag magas és alacsony bitekre. Ezt követően elsőként a csempe alacsony 8 byte-ját tároljuk el majd ezt követően a magas nyolc byte-ot, így megkapjuk a fentebb kiszámolt 16 byte-os értéket. Ezáltal könnyel elérhetővé tettük a kép adatainkat a cpu számára, illetve a PPU-is helyesen képes ezeket megjeleníteni.

### 3.2.3. A név táblák (Name Tables) és tulajdonság táblák (Attribute Tables)

A név táblák alapvetően, ahogyan már fentebb olvashattuk egy-egy 32 x 30-as rácsként képzelhetők el, ahol minden egyes cella egy csempének felel meg (8 x 8 pixel). Ebből következően egy név tábla 256 x 240 pixelt tartalmaz, ez a CRT monitorok teljes képernyő területe. Egy rács elem, pedig egy, egy byte-os címet tartalmaz, amely az éppen aktív pattern tábla egy csempéjét címzi meg (mivel összesen 256 aktív csempénk lehet ezért elég egy byte a címzéshez).

Ahhoz, hogy a PPU-nk képes legyen egy játék során gyors és gördülékeny háttér változásokra több különböző eszköz fejlesztettek ki. Kezdve azzal, hogy a NES hardveren belül két név táblát helyeztek el, ezek okos címzésével és a hardver sajátosságainak kihasználásával (mirroring) képesek vagyunk az úgynévezett görgetés (scrolling) megvalósítására. Ez egyszerűen egy vertikális vagy egy horizontális finom lapozásnak írható le (egy-két sor pixel eltolásával, illetve megjelenítésével). Alapvető esetben a NES játékok vagy fix háttérrel rendelkeztek (lásd donkey kong), vagy csak horizontális vagy vertikális görgetést alkalmaztak (horizontális - Super Mario Bros.). Viszont a későbbiekben megjelentek komplexebb játékok is amelyek ezek vegyítését alkalmazták ilyen például a Metroid vagy a Legend of Zelda.

A mirroring az a jelenség, amikor két cím azonos memória területre mutat, ez azért fordulhat elő, mert a PPU nem teljes címzést használ (kevesebb bit-tel címez meg tartományokat). A név táblák esetében például egy 2 x 2-es rácsot szoktak képezni ennek segítségével, így kiterjesztve a két névtáblánk címzési tartományát. De ez a jelenség ez egész PPU memória felépítése során megfigyelhető.

| Memória címek   | Méret  | Leírás                                 |
|-----------------|--------|----------------------------------------|
| \$0000 - \$0FFF | \$1000 | Pattern tábla 1                        |
| \$1000 - \$1FFF | \$1000 | Pattern tábla 2                        |
| \$2000 - \$23FF | \$0400 | Név tábla 1                            |
| \$2400 - \$27FF | \$0400 | Név tábla 2                            |
| \$2800 - \$2BFF | \$0400 | Név tábla 3                            |
| \$2C00 - \$2FFF | \$0400 | Név tábla 4                            |
| \$3000 - \$3EFF | \$0F00 | A \$2000 - \$2EFF címterület mirror-ja |
| \$3F00 - \$3F1F | \$0020 | Szín paletta RAM indexek               |
| \$3F20 - \$3FFF | \$00E0 | A \$3F00 - \$3F1F címterület mirror-ja |

### 3.1. táblázat. A PPU memória kezelése (14 bit címek) mirroring jelenséggel

Minden egyes névtábla végén egy kisebb extra memória terület található, amelyet tulajdonság táblának (vagyis Attribute Table-nek) nevezünk. Ez egy kisebb táblázatként képzelhető el ahol minden egyes cellában egy byte adatot tárolunk, viszont ennek feloldása

bonyolultabb mint névtáblák esetében. Ez a 8 bit egy 4 X 4 csempéyi háttér terület szín palettáját határozza meg a következő képen:

- az első két bit a bal felső 2 x 2 csempe palettáját határozza meg,
- a második két bit a jobb felső 2 x 2-es terület palettáját határozza meg,
- a harmadik két bit a bal alsó 2 x 2-es terület palettáját határozza meg,
- végül pedig az utolsó két bit a jobb alsó terület színérért felelős.

Így a teljes képünket 8 x 8 ilyen byte-tal írhatjuk le.

### 3.2.4. Objektum Attribútum Memória (OAM)

Ez a memória terület 64 külön álló sprite tárolására képes. minden egyes OAM spritehoz négy byte adat tartozik. Az első byte meghatározza a vertikális vagy y koordinátáját a sprite-nak. A második byte azt határozza meg, hogy melyik 8 x 8-as csempe legyen megjelenítve az éppen aktív Pattern táblából. A harmadik byte segítségével különböző tulajdonságait vagyunk képesek befolyásolni a sprite-nak. Végül pedig az utolsó byte a horizontális vagyis x koordinátáját határozzam meg az objektumnak.

Az előbb bemutatott byte-ok közül az egyetlen bonyolultabb működésű a harmadik, ebben az estben is a különböző bit-ek más és más működést kódolnak. Az nulladik és az első bit egy előtér szín palettát választanak ki a sprite számára. A következő három bit (3, 4 és 5) nem használtak. Ezt követően az ötödik bit határozza meg, hogy a sprite a háttér elé vagy mögé kerüljön (ha értéke nulla akkor a háttér fölé fog kerülni, ha pedig egy mögé). Végül pedig a hatodik illetve hetedik bitek azt határozzák, hogy a sprite pixel-ei horizontálisan vagy vertikálisan helyet cseréljenek (tükrözve legyenek), nulla esetén a sprite eredeti formájában marad, egy esetén, pedig meg tükröződik. Ez egy nagyon hasznos tulajdonság, hiszen így a játék fejlesztők rengeteg memória területet spórolhattak a Pattern táblákból. Mivel több olyan objektumot, karaktert vagy effektet is tervezhettek amelyek vagy horizontálisan vagy vertikálisan szimmetrikusak voltak (erre az egyik legjobb példa a felvétő gomba a Super Mario Bros. videojátékból).

### 3.2.5. PPU hardveres hibái (bug-ok)

Mivel hardveres emulálást készítünk, ezért nem lehetünk el az eredeti hardver hibái mellett. Az esetek többségében a NES játékok fejlesztői kihasználták ezeket az hazárdokat a játékfejlesztések során. Tehát ha az eredeti játékokkal szeretnénk játszani ezekkel is maradéktalanul meg kell ismerkednünk.

A PPU leghíresebb ilyen hibája, a sprite túlcordulás kezelése (Sprite Overflow Bug). Ez a OAM feldolgozása és megjelenítése közben alakulhat ki a NES-ben. Ennek megisméréséhez először is érdemes áttekinteni, hogy a PPU milyen állapotgép alapján dolgozza fel a OAM-ot. A PPU-n belül található egy másodlagos OAM amely nyolc sprite eltárolására képes, ennek a nyolc sprite-nak a megjelenítése történik a NES egy 256 pixeles sorban. minden egyes sorral ebbe a memória területbe töltjük be az éppen aktív sprite-okat, ennek következtében a 64 sprite közül csupán nyolcat tudunk egy sorba megjeleníteni. Ezek alapján az OAM megjelenítés a következő négy lépésből áll:

1. először is megtisztítjuk a másodlagos OAM-ot,
2. majd végigvizsgáljuk a teljes OAM-ot és kiválasztjuk azokat a sprite-okat amiket a következő sorban meg kell jeleníteni (ezekből az első nyolcat),

3. ha a megtaláltuk a 8 megjelenítendő sprite-ot akkor sajnos egy hibás implementációval végig vizsgálja az eszköz, hogy van-e még aktív sprite a sorban (ha talál ilyet beállítja a Sprite Overflow Flaget),
4. végül pedig a PPU feltölti a megjelenítéshez szükséges regisztereket az éppen aktív spritok adataival.

A következő sorban megjelenítendő aktív sprite-ok kiválasztása, az objektumok y értéke alapján történik az OAM-ban tárolt prioritási sorrend alapján. A fent említett rendszerhiba a harmadik lépésben következhet be, amikor a PPU megtalálta a nyolcadik sprite-ot, ezt követően elkezdi vizsgálni a maradék OAM területet, viszont ennek vizsgálata csak az első esetben következik be helyes tulajdonság szerint (y érték). Ezt követően a sprite-ok tulajdonságaiban diagonálisan haladunk (tehát a második keresés a Pattern tábla cím alapján történik, és későbbiekben így tovább halad az objektum négy byte-ján keresztül), így olyan esetben is bejelezhet a sprite túlcsordulást jelző flag (Sprite Overflow Flag), amikor ez nem is történt meg. Erre a hibára több híres játék is épített az egyik leghíresebb a The Legend of Zelda, de például a Ninja Gaiden és Castlevania sorozatokban is meg jelent.

Egy másik kevésbé ismert hazárd az OAMADDR regiszterrel kapcsolatban találtak meg. Ez általában akkor jött elő amikor nem a DMA vezérlőt használták az OAM frissítésére, hanem a szimpla byte elérését. Ilyenkor néha elő fordult, hogy egy bájtot rossz OAM területre másolt az eszköz, ezzel beszennyezve az OAM-ot. Ennek a bug-nak későbbi kompatibilitási okai voltak.

### 3.3. 2A03 a NES fő vezérlő egysége

A NES videojáték konzol egy komplex több feladatot ellátó fő végrehajtó egységgel rendelkezik, melynek neve 2A03 (RP2A03[G] NTSC konzolokban). A chip három fő hardveres elemből áll. Először is a kor legjobban elterjedt CPU-jának a MOS 6502-esnek egy módosított változatát tartalmazza, emellett helyet kap még az APU tárprocesszor (co-processzor) is, amely a hang generálásért felelős végrehajtó egység, illetve az adatok gyorsabb másolását elősegítő DMA egységet. A következőkben ezt a három hardveres elemet ismerhetjük meg kicsit részletesebben.

#### 3.3.1. MOS 6502 - központi feldolgozó egység (Central Processing Unit, CPU)

A CPU alapja a MOS Technology 6502-es 8 bites architektúrával rendelkező mikroprocesszora. A processzort Chuck Peddle és Bill Mensch amerikai mérnökök terveztek és először 1975-ben mutatták be. Már a kezdetektől nagy sikernek számított kompakt és egyszerű designja és jó programozhatósága miatt. Egyszerűsége miatt sokkal kevesebb tranzisztorban elfért (3510 tranzisztor) mint vetélytársai az Intel és Motortollától. Anyagköltsége miatt nagyon kedvező áron kezdhették el forgalmazni (csupán 25 dollár). A leghíresebb felhasználásai az Apple 1, 2, illetve a Commodore 64. Több játék konzol is felhasználta a chip architektúráját saját CPU-ik kialakításához (NES, SNES, Atari 2600).

A processzor egy órajel bemenettel rendelkezik  $\phi_0$ , amelyet a NES esetén egy külső kvarc kristály hajt meg 1.789773 MHz frekvencián (3.2). Ebből a beérkező órajelből állít elő a processzor két, eltolt fázisú órajelet  $\phi_1$  és  $\phi_2$ . Ezzel több szinkronizációs lehetőséget engedve egy órajelenbeliül a külső hardvereknek. A 6502 8 bites architektúrája azt jelenti, hogy a processzorunk egy byte adatot képes feldolgozni/módosítani egy órajel lefutása alatt. Tehát 8 bites adatbusszal és egy 16 bites címbusszal rendelkezik. Az 2A03-as model-

leken a Nintendo mérnökei annyiban módosították az eredeti MOS architektúrát, hogy letiltották a cpu módját.



3.8. ábra. A MOS 6502 egyedi nyolcbites architektúrája

A 6502 8 bites architektúrája hat regiszterrel rendelkezik. Ebből három (A, X, Y) általános programozási célokot tölt be. Hárrom pedig speciális belső információk tárolására szolgál (PC, SP, SR). Ezek elhelyezkedését és logikai kapcsolatát láthatjuk a 3.8 képen.

1. *Program számláló (Program Counter, PC):* 16 bites regiszter, amely a program jelenlegi címét tartalmazza (ennek a regiszternek a mérete határozza meg a cím tartományt)

2. *Verem mutató (Stack Pointer, S)*: 16 bites regiszter, viszont felső nyolc bitje fix decimális egy (0000 0001) értékkal rendelkezik. Tehát valójában egy 8 bites regiszter, amely a verem aktuális címére mutat.
3. *X index regiszter*: 8 bites index regiszter programozás során fölfelé és lefelé is számolhatunk vele. Általában adat indexelésre használjuk, de aritmetikai művelet is végrehajtható rajta.
4. *Y index regiszter*: 8 bites index regiszter használata megegyezik a az X index regiszter használatával.
5. *Akkumulátor (Accumulator, A)*: 8 bites regiszter fő szerepe az adat tárolás, ha bármilyen műveletet végzünk a processzoron annak eredménye ebben a regiszterben kerül eltárolásra.
6. *Processzor Státusz Regiszter (P)*: 8 bites regiszter, a processzor státusz flagjeit tárolja a 3.9 ábrán látható módon.



**3.9. ábra.** A MOS 6502 processzor státusz regisztere (P)

Az eredeti MOS 6502 tizenhárom címzési móddal rendelkezett, többek között akkumulátoros, beleértett (implied), közvetlen (immediate), relatív, abszolút X és Y szerinti, ezek között voltak ritkábban és gyakran használt címzési módok is. Nem minden címzési módban van lehetőségünk az összes utasítás használatára.

A processzorunk utasítás készlete 56 alapértelmezett utasításból állt. Az utasítások műveleti kódjai egy bájtosak és a különböző címzési módokkal 151-nyi kód helyet foglalnak el a rendelkezésre álló 256-ból. A maradék fennálló utasítás helyet általában a processzor gyártók foglalt (reserved) utasítás területékként tartják fent. Ezek használata során előre nem definiált jelenségek történnek a 6502-ben. Ez azért fontos a NES szempontjából, mert a játék fejlesztők az 2A03-ra történő fejlesztés során többször is használtak, ilyen úgy nevezett illegális utasításokat a program ROM helytakarékkossága szempontjából. Ilyen például a Beauty and the Beast (E) (1994) amely a \$80 műveleti kódot használja (ez egy 2-bájtos NOP utasításnak felel meg) vagy a F-117A Stealth Fighter amely a \$89-ot (hasonlóan egy 2-bájtos NOP). Ahhoz, hogy teljes hardveres emulálást készítsünk ezeket az illegális utasításokat is le kell implementálnunk. Az 3.10 ábrán a 2A03-ban található 6502 teljes utasítás készlete látható. A táblázat horizontális tengelye a 8 bites műveleti kódok ofszetje, a vertikális pedig a kezdő címe (így a oszlop és a sor érték összegéből megkaphatjuk az opkódot). A piros utasítások a program végrehajtást befolyásolják, a zöldök a különböző ALU utasítások, a kékek pedig az úgynevezett olvasási és írási utasítások (read-modify-write, RMW)). Az összes szürke utasítás pedig illegális utasítás, ezek többsége a kék és zöld utasítások valamilyen kombinációja. A táblázatban még az is látható, hogy az adat utasítás milyen címzési módban használható, illetve milyen adatokat várnak el bemenetként.

|     |     |     |     |     |       |       |     |     |       |       |     |     |     |     |     |     |     |       |       |     |       |       |       |     |       |       |     |     |     |     |     |
|-----|-----|-----|-----|-----|-------|-------|-----|-----|-------|-------|-----|-----|-----|-----|-----|-----|-----|-------|-------|-----|-------|-------|-------|-----|-------|-------|-----|-----|-----|-----|-----|
| +00 | +04 | +08 | +0C | +10 | +14   | +18   | +1C | +01 | +05   | +09   | +0D | +11 | +15 | +19 | +1D | +02 | +06 | +0A   | +0E   | +12 | +16   | +1A   | +1E   | +03 | +07   | +0B   | +0F | +13 | +17 | +1B | +1F |
| 00  | BRK | NOP | PHP | NOP | BPL   | NOP   | CLC | NOP | ORA   | ORA   | ORA | ORA | ORA | ORA | ORA | STP | ASL | ASL   | ASL   | STP | ASL   | NOP   | ASL   | SLO | SLO   | ANC   | SLO | SLO | SLO | SLO |     |
| d   | a   | *+d | d,x | a,x | (d,x) | d     | #i  | a   | (d),y | d,x   | a,y | a,x | d   | a   | a   | d,x | a,x | (d,x) | d     | a,x | (d,x) | d     | #i    | a   | (d),y | d,x   | a,y | a,x |     |     |     |
| 20  | JSR | BIT | PLP | BIT | BMI   | NOP   | SEC | NOP | AND   | AND   | AND | AND | AND | AND | AND | STP | ROL | ROL   | ROL   | STP | ROL   | NOP   | ROL   | RLA | RLA   | ANC   | RLA | RLA | RLA | RLA |     |
| a   | d   | a   | *+d | d,x | a,x   | (d,x) | d   | #i  | a     | (d),y | d,x | a,y | a,x | d   | a   | a   | d,x | a,x   | (d,x) | d   | a,x   | (d,x) | d     | #i  | a     | (d),y | d,x | a,y | a,x |     |     |
| 40  | RTI | NOP | PHA | JMP | BVC   | NOP   | CLI | NOP | EOR   | EOR   | EOR | EOR | EOR | EOR | EOR | STP | LSR | LSR   | LSR   | STP | LSR   | NOP   | LSR   | SRE | SRE   | ALR   | SRE | SRE | SRE | SRE |     |
| d   | a   | *+d | d,x | a,x | (d,x) | d     | #i  | a   | (d),y | d,x   | a,y | a,x | d   | a   | a   | d,x | a,x | (d,x) | d     | #i  | a     | (d),y | d,x   | a,y | a,x   |       |     |     |     |     |     |
| 60  | RTS | NOP | PLA | JMP | BVS   | NOP   | SEI | NOP | ADC   | ADC   | ADC | ADC | ADC | ADC | ADC | STP | ROR | ROR   | ROR   | STP | ROR   | NOP   | ROR   | RRA | RRA   | ARR   | RRA | RRA | RRA | RRA |     |
| d   | (a) | *+d | d,x | a,x | (d,x) | d     | #i  | a   | (d),y | d,x   | a,y | a,x | d   | a   | a   | d,x | a,x | (d,x) | d     | #i  | a     | (d),y | d,x   | a,y | a,x   |       |     |     |     |     |     |
| 80  | NOP | STY | DEY | STY | BCC   | STY   | TYA | SHY | STA   | STA   | NOP | STA | STA | STA | STA | NOP | STX | TXA   | STX   | STP | STX   | TXS   | SHX   | SAX | SAX   | XAA   | SAX | AHX | SAX | TAS | AHX |
| #i  | d   | a   | *+d | d,x | a,x   | (d,x) | d   | #i  | a     | (d),y | d,x | a,y | a,x | d   | a   | a   | d,y | a,y   | (d,y) | d   | #i    | a     | (d),y | d,y | a,y   | a,y   |     |     |     |     |     |
| A0  | LDY | LDY | TAY | LDY | BCS   | LDY   | CLV | LDY | LDA   | LDA   | LDA | LDA | LDA | LDA | LDA | LDX | LDX | TAX   | LDX   | STP | LDX   | TSX   | LDX   | LAX | LAX   | LAX   | LAX | LAX | LAS | LAX |     |
| #i  | d   | a   | *+d | d,x | a,x   | (d,x) | d   | #i  | a     | (d),y | d,x | a,y | a,x | #i  | d   | a   | d,y | a,y   | (d,x) | d   | #i    | a     | (d),y | d,y | a,y   | a,y   |     |     |     |     |     |
| C0  | CPY | CPY | INY | CPY | BNE   | NOP   | CLD | NOP | CMP   | CMP   | CMP | CMP | CMP | CMP | CMP | NOP | DEC | DEX   | DEC   | STP | DEC   | NOP   | DEC   | DCP | DCP   | AXS   | DCP | DCP | DCP | DCP |     |
| #i  | d   | a   | *+d | d,x | a,x   | (d,x) | d   | #i  | a     | (d),y | d,x | a,y | a,x | #i  | d   | a   | d,x | a,x   | (d,x) | d   | #i    | a     | (d),y | d,x | a,y   | a,x   |     |     |     |     |     |
| E0  | CPX | CPX | INX | CPX | BEQ   | NOP   | SED | NOP | SBC   | SBC   | SBC | SBC | SBC | SBC | SBC | NOP | INC | NOP   | INC   | STP | INC   | NOP   | INC   | ISC | ISC   | SBC   | ISC | ISC | ISC | ISC |     |
| #i  | d   | a   | *+d | d,x | a,x   | (d,x) | d   | #i  | a     | (d),y | d,x | a,y | a,x | #i  | d   | a   | d,x | a,x   | (d,x) | d   | #i    | a     | (d),y | d,x | a,y   | a,x   |     |     |     |     |     |

**3.10. ábra.** A 2A03-ban található 6502 teljes utasítás készlete

Az 3.1.1 fejezetben már olvasottak alapján a 2A03 chip esetén is foglalkoznunk kell a memory mapped I/O architektúrával, tehát a különböző hardveres komponensek elérésével. A 6502, 16 bites címtartománnal rendelkezik, ez pedig a NES tervezői 3.2 táblázatban látható módon osztották fel. Itt is jól megfigyelhető a PPU-ban már bemutatott tükrözés (mirroring) jelensége.

| Cím tartomány   | Méret  | Eszközök                                                                                    |
|-----------------|--------|---------------------------------------------------------------------------------------------|
| \$0000 - \$07FF | \$0800 | 2 KB belső memória                                                                          |
| \$0800 - \$0FFF | \$0800 | \$0000 - \$07FF tükröképe (Mirrors)                                                         |
| \$1000 - \$17FF | \$0800 | \$0000 - \$07FF tükröképe (Mirrors)                                                         |
| \$1800 - \$1FFF | \$0800 | \$0000 - \$07FF tükröképe (Mirrors)                                                         |
| \$2000 - \$2007 | \$0008 | PPU regiszterei                                                                             |
| \$2008 - \$3FFF | \$1FF8 | \$2000 - \$2007 tükröképei (Mirrors) 8 bájtonként ismételve                                 |
| \$4000 - \$4017 | \$0018 | APU és I/O regiszterek                                                                      |
| \$4018 - \$401F | \$0008 | Olyan APU és I/O funkciók amik alapértelmezetten ki vannak kapcsolva                        |
| \$4020 - \$FFFF | \$BFE0 | Játék kártya területek, először Mapper regiszterek, ezt követően program ROM és Program RAM |

**3.2. táblázat.** A CPU memória címtartománya (16 bit címek) mirroring jelenséggel

### 3.3.2. Audió feldolgozó egység (Audio Process Unit, APU)

Az APU is egy PPU-hoz hasonló társprocesszor (coprocessor), tehát a CPU a chip belső regiszterein keresztül képes elérni és változtatni működését (a CPU adatbuszon keresztül 3.2). Az eszköz belsejében öt digitális jel generátor található, amelyek szimultán működnek a chip belsejében. A regiszterek segítségével ezeknek a jelgenerátoroknak a működését tudjuk befolyásolni (frekvencia, amplitúdó (milyen hangos az adott csatorna), illetve az adott csatorna ki és bekapcsolása). A csatornák részletesebb vizsgálatához tekintsük meg a 3.11 ábrát.



**3.11. ábra.** Az audió feldolgozó egység hang sávjai

1. *Első négyzetgel generátor kimenete (pulse<sub>1</sub>):* Az első és második pulzus generátor a NES melodikus dallamaiért volt felelős. A két pulzus jel generátor a korabeli számítógépek jellegzetes Y hangjára hasonlított.
2. *Második négyzetgel generátor kimenete (pulse<sub>2</sub>):* A pulzus generátorok felhasználása megegyezik az első pulzus generátoréval.
3. *Háromszöggel generátor:* A legtöbb esetben a basszus hangok képzésére használták. Itt érdekes még megfigyelni, hogy a háromszög generátor nagy lépésekben dolgozik a 3.11 ábrán is jól láthatók ezek (összesen 32 lépéssel rendelkezik, amely minden egyikhez 4 bites értéket rendel). Illetve egy szinusz jelet próbál generálni a nagyobb lépésekkel (ebből csak fűrész jel lesz). Ezzel egy enyhe, tompa hangokat tartalmazó basszust nyújt a NES játékoknak.
4. *Zaj generátor:* Ez a csatorna egy fehér zaj generátor, ez a korabeli TV-k jel nélküli (no-signal) hangjához hasonlított leginkább. A hang mérnökök ezt a zajt ütőszökhöz és dobokhoz használták legtöbbször.
5. *Delta modulációs csatorna:* Ez a hangcsatorna egy delta-kódolt egy bites kimenetet működtet vagy a hardverben található hét bites számlálót töltetheti. A kimeneten képes előre beállított DPCM minták lejátszására (ez egy veszteség mentes kódolási típus). Ennek segítségével letudtak játszani beszédhangokat, dallamokat, ütőszöket. Viszont ügyelni kellett rá, hogy csak rövid idő tartalmú minták lejátszása volt lehetséges a hardverben található puffer mérete miatt. Használata befolyásolta az OAM DMA működését.

Az öt szimultán működő jel generátorak egy-egy digitál-analóg konverter változtatja analóggá a kimenetét. A konverterek egymástól nem lineárisan függő elemek, amik befolyásolhatják az egyes csatornák amplitúdóját. Ezeket a jeleket az APU analóg keverője alakítja egy analóg kimenetté a későbbiek során. A fentebbekben leírt nem linearitást és az analóg csatornák keverését a 3.1, 3.2 és 3.3 egyenletekkel tudjuk legjobban közelíteni.

$$Output = Pulse_{out} + Tnd_{out} \quad (3.1)$$

$$Pulse_{out} = \frac{95.88}{\frac{8128}{pulse_1+pulse_2} + 100} \quad (3.2)$$

$$Tnd_{out} = \frac{159.79}{\frac{1}{\frac{triangle}{8227} + \frac{noise}{12241} + \frac{dmc}{22638}} + 100} \quad (3.3)$$

### 3.3.3. Direct Memory Access - OAM DMA

A 2A02-on belül egy korai DMA megvalósítás is helyet kapott. Ennek szerepe az adatok fix RAM területre való másolása, gyorsabban mintha a 6502-nek kellene ezzel foglalkoznia. A CPU-nak négy órajel ciklusra lenne szüksége egy adat kiolvasására és másolására, a DMA ezt fele annyi idő alatt (két órajel ciklus) eltudja végezni, viszont csak előre fixált területről tudunk olvasni és írni. Ez a modern DMA megvalósításokhoz képest egy egyszerűbb és primitívebb eszköz, mivel amikor ezen keresztül másolunk a DMA elveszi a CPU adatbuszát és ready jelét így a CPU egy blokkolt állapotba (stand by) a másolás idejére.

Napjainkban a DMA-k már nem blokkolják a mikrokontrollerek/processzorok működését, így nem veszítve CPU számítási kapacitást.

A NES esetén a DMA szerepe az Objektum Attribútum Memória gyors feltöltése a mozgó csempe adatokkal. minden esetben ha a DMA-t aktiváljuk (belő regiszterén keresztül) a teljes OAM RAM feltöltésre kerül. Egy írási/olvasási ciklus során, a CPU által beállított WRAM címről (általában \$0200), egy bájtnyi adatot olvas az eszköz. Majd az olvasott adatot a CPU címtartományában található \$2004-es PPU OAMDATA regiszterbe írja. Ezt a ciklust ismétli (növelve a WRAM címszámláló értékét) a DMA, mindaddig amíg a teljes memória végére nem érünk.

Ez a hardveres komponens szoros kapcsolatban áll az APU DMC csatornájával. Mivel ez a csatorna képes szüneteltetni DMC elérés során a DMA másolását négy órajelciklus idejéig.

### 3.4. NES kontroller

A Nintendo Entertainment System videó játék konzolhoz különlegesebbnél különlegesebb bemeneti eszközök születtek az évek során. Ebben a dokumentumban viszont a modern értelemben vett a kontrollerek atyját szeretném jobban bemutatni. Ez pedig a NES gamepad vagy NES standard kontroller néven vált ismerté a 3.1 ábrán is ez látható.

Az eszköz működése rendkívül egyszerű, hardvere csupán egy paralel-soros léptető (shift) regiszterből áll. Összesen nyolc gomb található a kontrolleren, ezt nyolc gombot mintavételezi a CPU (a címtartományában található regiszteren keresztül 3.2.). A mintavételezés pillanatában a lenyomott gombok nullás, a többi gomb pedig egyes értékkel szerepelnek a léptető (shift) regiszter bemenetén. A mintavételezést követően pedig a 2A03-tól kapott órajel hatására, bitenként kiolvasásra kerül a mintavételezett érték. A két kontrollerből beérkező értékek még egy hardveres negáláson esnek át az alaplapon 3.1, a könnyebb feldolgozhatóság érdekében.

## 4. fejezet

# NES FPGA alapú újra gondolása

A diplomaterv projektem egyik fő célja, hogy egy hardveres emulátort készítsek a Nintendo Entertainment System játékkonzolhoz. Alapvetően emulátort lehet készíteni szoftveres illetve hardveres módon. A szoftveres emulátorok alapja az eredeti hardver szimulálása egy magasabb programozási nyelven íródott szoftver segítségével (python, java), viszont ezek az emulátorok az esetek többségében nem tudják az eredeti hardver időzítésit pontosan betartani, ezért nem teljesen autentikus a játék élmény. Hardveres emulálásnál általában egy FPGA chipet használnak és ebben implementálják az eredeti hardveres működéseket. Az általunk választott megvalósítás sokkal idő igényesebb, viszont ennek segítségével képesek vagyunk az eredeti eszköz leg pontosabb emulálására. Illetve egy erősebb FPGA segítségével modernizálhatunk egy régebbi konzolt is, persze bármilyen hardveres módosítás esetén (amely eltér az eredeti eszköztől), át kell gondolnunk, hogy a meg változott körülmények között is képesek leszünk-e futtatni az eredeti szoftvereket a hardveren.

Ebben a fejezetben azt fogom bemutatni, hogy milyen hardveres változtatásokat/fejlesztéseket eszközöltem, az eredeti NES-hez képest, hogy egy friss, modernebb megjelenést adjak az eredeti konzolnak.

### 4.1. Képalkotás

Már az előző 3.2.1 fejezetben olvashattuk, hogy a NES egy kompozit jelet állított elő CRT televíziók számára. Ez talán a játék konzol legelavultabb része, hiszen napjainkban már ezt a televízió típust nem is lehet beszerezni. Ezeket teljes mértékben leváltották a VGA alapú (DVI, HDMI, DisplayPort csatlakozóval ellátott) különböző méretű és felbontású lapos TV-k. Ezeknek a megvalósításoknak rengeteg előnye van az analóg megjelenítéssel szemben. Az egyik legjelentősebb, hogy ezen keresztül képesek vagyunk torzításmentes átvitelre.

A NES eredeti felbontása 256 x 240 pixel és 60 Hz (NTSC). Ez a méret sajnos nem felel meg a modern VGA szabványoknak, úgy hogy ezen a téren módosítanunk kell a PPU képgenerálásán. A módosítások során a VGA adatok generálását helyeztem elő térbe. A legkisebb VGA kép méret amivel érdemes dolgozni és a modern TV-k és monitorok támogatnak az a 640 x 480 pixel és 60 Hz, szerencsénkre ebbe a méretbe pontosan elfér a kétszeres NES kép méret. Ebből eredeztetve, ha egy pixel-t 2 x 2 pixel-el reprezentálunk, akkor 512 x 480 pixel méretű képet kapunk ennél bonyolultabb megoldás is létezik (bilineáris interpoláció), de ez a legegyszerűbb ezért kezdetben ezt implementálom. Ez a változtatás lehetővé teszi, hogy NES nyomtatott huzalozott kártyáján modern HDMI csatlakozót helyezzék el.

Illetve ebből a módosításból következőleg a rendszerünk működési frekvenciája is meg változik, a VGA jelünk pixel óra jele 25 Mhz lesz, az RGB adatok pedig 250 Mhz-el lesznek

továbbítva a TV vagy monitor felé. Ez azt jelenti, hogy a PPU-nk működési órajelét is meg kell változtatni (eredetileg körülbelül 5.37MHz NTSC modellben). A NES nyomtatott huzalozott kártyáján ezért egy magasabb órajelforrást kell elhelyezni. Az órajel pontos beállítására pedig az FPGA chip DCM (Digital Clock Manager) és PLL (Phase Locked Loop) funkcióit használhatjuk.

## 4.2. Audio

Az audió feldolgozó egység (Audio Process Unit, APU), egy öt külön álló hangsávot kezelő analóg komponens, mely a NES eredeti hangját, nem lineáris keverés segítségével állította elő ezekből a csatornákból. Ezt a jelet a hardveres emulálás során mi egy digitális jelként állítjuk elő az FPGA-val. Ahhoz, hogy eredeti eszköz hangját élvezni tudjuk ezt először egy DAC segítségével analóg jelékké kell konvertálnunk, végül pedig egy megfelelő erősítőn át egy Jack csatlakozóra kivezethetünk. Az erősítő típusa attól függ, hogy fülhallgatón vagy pedig hangszórón szeretnénk, hogy ezek az ikonikus dallamok megszólaljanak. Egy jó kompromisszumos megoldás a kártyát fülhallgató erősítővel ellátni mivel hangszórókból léteznek olyan modellek (belül erősítővel rendelkezők), amelyek ezzel az gyengébb erősítővel is működnek (így minden a két opció fennáll a konzol használatra). Az eredeti konzol mono hangzással rendelkezett viszont mi ezt jelet a DAC működése miatt, minden a két fülre kivezethetjük, ezzel sztereó hangzást imitálva (ettől még természetesen ez nem lesz valódi sztereó jel).

Mivel a NES nyomtatott huzalozott kártyáján már a képalkotás miatt elhelyezek egy HDMI csatlakozót, ezért egy kisebb I2C szint illesztő komponens segítségével a HDMI hang csatornái is bekötethetők. Így lehetővé téve, hogy egy esetleges későbbi fejlesztés során a TV/monitor beépített hangszóróján szólaljon meg játékkonzolunk.

## 4.3. Játékok tárolása

A NES játékok programkódja és Pattern tábla adatai az esetek többségében a játék kazetta saját program és karakter ROM-jában helyezkedtek el. Ez a megoldás rengeteg extra területet emészteni fel a nyomtatott huzalozott kártyán (a kazetta befogadó egységet rá kéne tervezni az eszközre), illetve az összes teszt játéknak fizikailag a birtokomba kell lennie ehhez (és egyéb hardveres teszteket nehezen lehetne megvalósítani).

Ennek következtében egy új megoldást kellett kitalálni arra, hogy az emulált NES hardvert megfelelő módon ellássam játékokkal. Az egyik kézen fekvő megoldás, hogy a hordozó kártyára el lehet helyezni egy statikus RAM modult amelynek elég nagynak kell lennie ahhoz, hogy a játék kártyák karakter és program ROM-ját egyszerre tartalmazza. A legnagyobb NES játék 768 KB memória területtel rendelkezett és ez a Kirby's Adventure volt, ehhez képest az összes többi játék 512 KB-os vagy ennél kisebb volt. A NES hardver indítás előtt ezt a SRAM-ot fel kell töltenünk a játékkal, majd a hardver indítását követően a PPU és CPU innen fog dolgozni.

Annak érdekében, hogy több játékot is képesek legyünk tárolni érdemes egy nagyobb méretű háttértárat is tervezni az eszközre, amely a játékokat fogja tartalmazni, erre ideális lehet egy SD kártya. A projekt kezdeti szakaszában ezt még nem használom, viszont az esetleges jövőbeli fejlesztések miatt érdemes, már most a kártyára tervezni egy ilyen olvasót.

A NES hardver játék megjelenítéshez szükséges belső memória területeit, pedig az FPGA-ba kialakítható Blokk/LUT ram-ba helyezhetjük helyezhetjük el (Név táblák, OAM, másodlagos OAM).

#### 4.4. Kompakt hordozható méret

A NES újra tervezésének egyik fő aspektusa a méret csökkentése. Az eredeti konzol 256 mm hosszú, 203.2 mm széles és 88.9 mm vastag volt. Ezt a modern nyomtatott áramkörök tervezésével, és a komponensek kis méretével sokkal kisebb területre csökkenhető. Ezzel kompakt hordozható kialakítást kölcsönözve a játék konzolnak.



4.1. ábra. A NES játék konzol dimenziói

A kompakt méret mellett szerettem volna megőrizni az eszköz nosztalgia faktorát, ezért a kártyámon elhelyeztem két eredeti NES játék kontroller csatlakozót (kooperatív játékok miatt). Ez természetesen az eredeti hardver kontrollereivel kompatibilis.

## 5. fejezet

# FPGA NES kártya ismertetése

A NES hardverének újragondolásából az 5.1 ábrán látható blokk diagramot készítettem, ez a nyomtatott huzalozott kártyák tervezésének első lépése. Már itt érdemes feltüntetni a különböző áramköri elemek közti kommunikációs utakat (busz típusokat), illetve ezek irányát. Ez alapján a diagram alapján, pedig elkezdődhet a különböző komponensek keresése, ezt követően át kell gondolnunk ezek fogyasztását és feszültség szintjeit ezekből az adatokból pedig megtervezhető a kártya táp ellátása is (a mi esetünkben már kiegészítettem ezzel a blokk diagramot).



5.1. ábra. NES kártya blokkdiagramja

A komponensek, közül az FPGA chip kiválasztása a legnehezebb, ennek menete általában az, hogy megpróbáljuk felmérni a hardverünk méretét és ez alapján választunk megfelelő méretű chip-et. A mi esetünkben az egyik legkomplexebb elem a 6502-es 8-bites processzor amely méretét az OpenCores weboldalon található nyílt forráskódú hardver tervek alapján megbecsülhetjük körülbelül 1000 LUT-ra. Mivel NES hardver működéséhez három fő komponens kell (CPU, APU, PPU) ezek méretét egyesével felülbecsülhetjük a legkomplexebb alkatrész méretével, így összesen 3000 LUT-ot kapunk. Ehhez még érdekes a VGA jel elő állítását (HDMI jel kódolása), illetve az audió jel kezelését még hozzá számolni, erre is jó felső becslés az 1000 LUT. Végül még érdemes tartalékkal is számolni ezért egy 5000-6000 LUT-al rendelkező FPGA chip valószínűleg elég nagy ahhoz, hogy a

teljes projekt elférjen benne. Fontos kritérium még, hogy DCM-el illetve PLL-el rendelkezzen a chip az egyedi órajelek előállítása érdekében (például a 250 MHz a VGA bit-ek kiadásához), ezen kívül még Blokk-RAM-ra is szükség lesz legalább akkorára mint a NES hardverének belső memóriája (például Név táblák 2 kilobájt).

A feladat megvalósításához egy Spartan-6-os Xilinx FPGA állt a rendelkezésemre, amely megfelel a fent említett összes elvárásnak. Ez a chip nagyban meggyorsította a nyáktervezés menetét is, mivel az egyetemi Spartan-6-os fejlesztő kártyák fő komponense is ez az FPGA volt (erről itt olvashatunk részletesen [8]). Ezt a fejlesztő kártyát vettem alapul a NES kártyám fejlesztő portjának kialakítása, az SPI flash bekötése, illetve a táp vonalak kialakítása során is.

A kártyán az alábbi komponensek találhatók:

- *FPGA*: Xilinx XC6SLX9-2TQG144C típusú Spartan-6-os FPGA, amely lehetővé teszi összetettebb logikák és mikroprocesszoros rendszerek megvalósítását. Az eszköz főbb jellemzői:
  - 5720 darab 6 bemenetű LUT és 11440 darab flip-flop
  - 32 darab 18 kilobites blokk-RAM
  - 16 darab DSP48A1 blokk (elő összeadó, 18 x 18 bites előjeles szorzó és akkumulátor)
  - 4 darab DCM (Digital Clock Manager) és 2 darab PLL (Phase Locked Loop) modul
- *Memóriák a program és az adatok tárolására*:
  - Egy 256k x 16 bites (512 kB), 10 ns-os aszinkron SRAM (Cypress CY7C1041DV33-10ZSXI)
  - Egy 32 Megabites SPI buszos soros FLASH memória (Atmel AT25DF321A), amely konfigurációs memóriaként is szolgál az FPGA számára
- *Egy MicroSD memóriakártya foglalat*:
  - Teljes MicroSD kártya protokoll
  - Egyszerű SPI protokoll
- *Beviteli eszközök*:
  - Két eredeti 7 lábas NES (GamePAD) kontroller csatlakozók
  - Reset és PROG gombok
- *Képfeldolgozás*:
  - HDMI csatlakozó
  - I2C szint illesztő (PCA9306DCUR típusú), a HDMI audió vonalainak illesztéséhez
- *Audio*:
  - Digitális-analóg átalakító (DAC), CS4334-KSZ típusú
  - 100mW Erősítő TS486IST típus (félhallgatókhöz)
  - CUI SJ1-3553NG 3.5mm Jack csatlakozó
- *Tápegységek*: MCP1612-ADJI/MS típusú szinkron Buck konverterek
- *Egy 50 MHz-es oszcillátor*
- *Csatlakozó a LOGSYS fejlesztői kábel számára*

## 5.1. Tápellátás

A tápellátás kialakítása a Logsys Spartan-6-os fejlesztői kártyáéhoz hasonló [8]. A NES kártya 5 V-os tágfeszültségről működik. Ezt a tápellátást, vagy a fejlesztő kábelről kapja az eszköz, vagy egy külső 5 V-os forrásból. A külső egyenfeszültségű forrás Shottky diódával védtem, illetve a kártyát töltést jelző zöld led-del is elláttam (PWR).

Az 5 V-os forrást két azonos típusú step down (Buck) konverterrel 3.3 V-ra és 1.2 V-ra konvertálom. Alapvetően az FPGA működéséhez kell a két feszültség szint. A 3.3 V az I/O vonalakért, DCM, PLL, és konfigurációért felelős, az 1.2 V pedig az FPGA belső magjának kell. Ezt a két tápvonalat az FPGA dokumentációja alapján (a táp lábaihoz közel) elláttam a megfelelő mennyiségű csatoló (coupling) és hidegítő (bulk) kapacitásokkal a stabil működés végett. A 3.3 V-ot a kártyán található többi alkatrész is használja (SRAM, MicroSDkártya, kontrollerek, a fejlesztő kábel is megkapja mint JTAG referencia feszültség ként stb.). Illetve a kártya a HDMI csatlakozó, DAC és erősítő komponensek esetén, a tápellátás 5 V-ját is felhasználja. A tápellátás schematik rajzát az F.3.1 függelékben láthatjuk.

## 5.2. Órajel források

A NES kártyán a Logsys fejlesztő kártyához hasonlóan, egy 50 MHz-es oszcillátort helyeztem el. Az FPGA, vagy a fejlesztő portról érkező CLK-től kapja az órajelét, vagy ezt az 50 MHz-es CLK-t használja. Ahhoz, hogy az FPGA használhassa ezeket az órajeleket egy-egy órajel bemeneti lábára (GCLK) kellett ezeket bekötni. Az oszcillátor segéd áramkörét az F.3.5 függelékben láthatjuk.

## 5.3. Memória - SRAM

A választott asszinkron SRAM mérete 256 k x 16 bit (byte-okban mérve 512 k), ez a méret megfelel a 4.3 fejezetben tárgyalt játék méreteknek (csak egy NES játék nem fog beleférni ebbe a RAM-ba). A választott memória előnye, hogy 10 ns elérési idejű asszinkron, statikus RAM, egyszerű kezeléssel. Ez azért jelent előnyt a konzol hardveres emulálása szempontjából, mert nem fogja ennek működését befolyásolni a memória elérési idő (a későbbiekben az elérést igazíthatjuk az általunk választott időzítéshez). DRAM esetén sokkal több időzítési paraméterrel kell számolni ez természetesen jóval megnehezíti a időzítés kritikus hardver létrehozását.

Az SRAM 18 bites címmel rendelkezik, amellyel megcímhetők a 2 byte-osával az adataink (256 k cím). A RAM-ból kiolvasott, illetve beírandó adatokat pedig a 16 bit-es adat vonalak olvasásával és írásával érhetjük el. Az SRAM vezérlésétől függően kiolvasható vagy írható egyszerre mind a 16 bit, de van byte-os elérési mód is.

Az SRAM szabványos vezérlési felülettel rendelkezik. Tehát következő vezérlő jelek segítségével tudjuk irányítani (ezek mindegyike negált logikájú): chip engedélyezés CSn, írás engedélyezése WEn, olvasás engedélyezés OEn, alsó byte engedélyezés LBn, végül pedig a felső byte engedélyezés UBn. A memória olvasási és írási idő diagramjait a [7] adatlapon olvashatjuk, kiegészítő áramkörét pedig az F.3.4 függelékben láthatjuk.

## 5.4. Digital Analog Converter és erősítő

A NES kártya egyetlen analóg alkatrészekre támaszkodó része a DAC-ot követő erősítő áramkör, ennek részletes megértéséhez tekintsük meg az 5.2 ábrát, amely az F.3.3 függelékből lett kiemelve.

A DAC komponenst egy 100MHz-en 600  $\Omega$ -as Ferrite Bead-el védem, az esetleges 5 V-os tápvonalról beszűrődő zajokkal szemben, ide szűrési okokból tantál és kerámia kondenzátorokat is helyeztem. Az alkatrész az FPGA által elő állított mono digitális "hang" jelet, átalakítja és kiadja mind a két kimenetén. Ezek az analóg jelek fognak az erősítést követően, a Jack csatlakozó bal és jobb fülhöz menő lábára csatlakozni.

A DAC két kimenetét, egy egy 3.3 uF-os csatoló kondenzátorral választom el az analóg résztől. Az áramkörben található C38, C39-es kondenzátorok és R27, R28-as ellenállások egy alul áteresztő szűrőt valósítanak meg. A kondenzátorok értékét, pedig a következő képlet alapján határoztam meg (az ellenállások értéke kötött volt az erősítő miatt):

$$C = \frac{R + 560\Omega}{4} * \pi * F_s * (R * 560\Omega) \quad (5.1)$$

Itt  $F_s$  az általunk választott audió jel frekvenciája az 560  $\Omega$  pedig a soros ellenállás értéke. Ezek alapján a két kondenzátorom értéke 3.3 nF vagy 2.7 nF lehet, mivel így 48 KHz-hez közeli értéket kapunk a frekvenciára. Az összes kerámia kondenzátornak, amely az analóg áramkör része NP0 (vagy C0G) dielektrikummal kell rendelkeznie, mivel ezeknek nincs piezoelektromos tulajdonságuk.



5.2. ábra. NES audió jelért felelős áramkörök

Az erősítő komponensnek egy fázis fordító erősítőt választottam, amely erősítése az R24 és R38 ellenállásokkal módosítható a következő képletek alapján:

$$Gain_{LINEAR} = -\frac{R_{FEED}}{R_{IN}} \quad (5.2)$$

$$Gain_{dB} = 20 * \lg\left(\frac{RFEED}{RIN}\right) \quad (5.3)$$

Az áramkörben jelenleg nem állítottam be erősítést az RFEED és RIN ellenállások értékét 20 kΩ-nak határoztam meg. Ez természetesen a hardveres tesztelés során cserélhető és állítható.

## 5.5. HDMI és I2C szint illesztő

A NES nyomtatott huzalozott kártyáján elhelyeztem egy HDMI csatlakozót és a körülötte elhelyezkedő áramkör segítségével felkészítettem VGA jelek kiadására. A teljes áramkör sematikus ábráját (schematic) az F.3.2 függelékben láthatjuk.

A jövőbeli fejlesztések miatt elhelyeztem a nyákon még egy I2C jel szint illesztőt is, amely segítségével az FPGA 3.3 V-on működő lábait, a HDMI audió jel kiadásáért felelős lábaihoz (SCL/SDA) illesztettem. Így a kártya támogatja a TV-k és monitorok beépített hangszóróit is.

Az alábbi ábrákon láthatjuk és olvashatjuk egy HDMI anya aljzat pin és láb kiosztását:



**5.3. ábra.** aljzat anya

| Funkció           | Láb | Funkció           | Láb |
|-------------------|-----|-------------------|-----|
| TMDS Data2+       | 1   | TMDS Clock Shield | 11  |
| TMDS Data2 Shield | 2   | TMDS Clock-       | 12  |
| TMDS Data-        | 3   | CEC               | 13  |
| TMDS Data1+       | 4   | Reserved          | 14  |
| TMDS Data1 Shield | 5   | SCL               | 15  |
| TMDS Data1-       | 6   | SDA               | 16  |
| TMDS Data0+       | 7   | DDC/CEC Ground    | 17  |
| TMDS Data0 Shield | 8   | +5 V Power        | 18  |
| TMDS Data0-       | 9   | Hot Plug Detected | 19  |
| TMDS Clock+       | 10  |                   |     |

**5.1. táblázat.** HDMI lábkiosztás

Mivel az FPGA nem minden I/O láb pára képes differenciális jelek küldésére, ezért figyelni kell, hogy a csatlakozót az FPGA melyik oldalához közel helyezem el. A HDMI szabvány az adat vonalak között 100 Ω-os impedancia különbséget ír elő (15% os toleranciával), ezért érdemes ezeket az vezetékeket minél rövidebben/egyszerűbben megoldani (FPGA bekötési oldalhoz közel).

Az alkatrészeim védelmére a HDMI 5 V-os tápellátását egy 100 mA-es biztosítékon (poli fuse) vezettem keresztül, ez túláram esetén véd. A csatlakozó fém burkolatát egy 1 MΩ-os ellenállással és egy 1 nF-os (1 kV-os) kapacitással földeltem, ez HDMI kimenetek esetén ideális.

## 5.6. A kártya bemenetei

A NES nyomtatott huzalozott kártyáját az eredeti játékkonzol kontroller csatlakozóival lát-tam el. Az eredeti kontroller a NES 5 V-járól működik, viszont a benne található parallel-soros átalakító (shift regiszter) adatlapja alapján 3.3 V-ról is működik. Ez azért fontos, mert így nem kell extra jelszint illesztő IC a kártyára, működtethetem az adat fogadást és az órajel küldést az FPGA I/O lábairól (3.3 V).

Ennek a kontrollernek egyedi hét lábas anya csatlakozója, van amit az 5.4 ábrán láthatunk.



**5.4. ábra.** NES kontroller csatlakozó

Ebből a hét lábból kettőnek csak rögzítési szerepe van, kettő a tápellátásért felelős és a maradék három lábon keresztül történik a kontrollerben található regiszter olvasása, vezérlése és a működési órajel küldése. A kártyára ebből a csatlakozóból kettőt helyeztem el a kooperatív játékok végett. A kontroller portok sematikus ábráját (schematic) az F.3.3 függelékben láthatjuk.

A pcb-n két gomb is helyet kapott az egyik az FPGA és ezáltal a NES reset gombja (RST), a másik pedig az FPGA újrakonfigurálását elindító nyomógomb (PROG). Az RST gomb pergésmentesítésért az FPGA a felelős. Ezek mindegyikét az F.3.5 függelékben láthatjuk.

## 5.7. MicroSD kártya

A MicroSD kártya csatlakozót teljes interfésszel tudtam implementálni, mivel az FPGA-nak még sok szabad I/O lába maradt. Ez azt jelenti hogy teljes SD kártya protokollt is megtudok valósítani a NES fejlesztő kártyán, az egyszerűbb soros SPI kommunikációt mellett/helyett. A MicroSD kártya segéd áramkörét az F.3.2 függelék sematikus rajzán láthatjuk.

Itt érdemes az áramkör ki-bekapcsolásáért felelős P-Mosfet-es áramkört megnézni, ez azért szükséges, mert a fent említett két protokoll közötti váltáshoz áramtalanítanunk kell az csatlakozót. A tápellátás elvétele mellett a felhúzó ellenállások tápellátását is elvesszük kikapcsolás során. Egyedül a CD kártya detektáló lábtól nem vesszük el, mivel ez csak azt jelzi, hogy van-e SD kártya a csatlakozóban (nem része a fent említett kommunikációs protokolloknak). Ezt be/ki kapcsolási eseményt az FPGA egy I/O lábának segítségével kontrollálhatjuk.

Az FPGA védelme érdekében elhelyeztem a az SD kártya CLK lábára egy  $33\ \Omega$ -os soros ellenállást, ezt a PCB layout tervezése során a lehető legközelebb helyeztem el az FPGA-hoz.

## 5.8. FPGA konfigurációs módok

A NES kártyának is, a LOGSYS Spartan-6 FPGA kártyához hasonló módon [8] két konfigurációs módja van. Az FPGA-t felprogramozhatjuk a fejlesztőportban található JTAG interféssz segítségével, illetve felkonfigurálhatja saját magát a kártyán található soros FLASH memóriából is. A konfigurációs módok között egy rövidzár segítségével (jumper) váltha-

tunk a LOGSYS-es kártyához hasonlóan. Ennek működését az 5.2 táblázatban olvashatjuk, illetve az F.3.6 függelékben láthatjuk.

| Jumper állása                                                                     | Konfigurációs mód | Leírás                                                                                                                                  |
|-----------------------------------------------------------------------------------|-------------------|-----------------------------------------------------------------------------------------------------------------------------------------|
|  | JTAG              | Az FPGA-t a JTAG interfacen keresztül kell felkonfigurálni.                                                                             |
|  | SPI               | Az FPGA az SPI buszos soros FLASH memóriából konfigurálja fel magát a tápfeszültség bekapcsolása vagy a PROG gomb megnyomását követően. |

### 5.2. táblázat. Fejlesztői port bekötése

## 5.9. Soros Flash memória

A NES kártyán egy Atmel AT25DF321A típusú, 32 Megabit-es SPI busszal rendelkező soros FLASH memóriát helyeztem el. Ez a komponens az FPGA számára konfigurációs memóriaként szolgál (tehát a NES hardverének binárisát fogja tartalmazni). Ennek a komponensnek az elhelyezése, bekötése és ezáltal működése is megegyezik a Logsys Spartan-6-os fejlesztői kártyán található Flash-el [8]. A soros Flash bekötését, pedig az F.3.4 függelékben láthatjuk.

## 5.10. LOGSYS fejlesztői port

A fejlesztői port kialakítása teljes mértékben megegyezik a Logsys fejlesztő kártyán található port-al, ennek köszönhetően az FPGA felprogramozása történhet a MIT tanszéken tervezett egyedi fejlesztő kábellel. Ennek részletes bemutatása a [8]-os dokumentáció része. A fejlesztői port kialakítását a ?? képen látható és FPGA bekötését pedig a ?? táblázatban olvasható, illetve az F.3.5 függelékben látható. Az FPGA sikeres felkonfigurálását egy zöld LED-el jelzem (DONE).

## 5.11. Nyomtatott áramköri terv

A schematik megalkotását követően, a pcb tervezés következő fázisa a layout (pcb rajzolat) elkészítése. Itt természetesen figyelembe kell vennünk az eddig meghatározott célokat 4.4, miszerint egy kompakt hordozható eszközöt tervezünk. Egy nyomtatott áramkör mérete nagyban függ a réteg felépítésétől, illetve a kiválasztott komponensek méretétől. Tehát minél több rétegből épül fel egy pcb és minél modernebb alkatrészeket használunk, annál nagyobb felületi alkatrész sűrűség érhető el. Természetesen ezekkel arányosán az ár is nő. Viszont mivel először egy prototípust fejlesztek, ezért egy kompromisszumos megoldást kellett választanom.

### 5.11.1. Réteg beállítások

A Logsys Spartan-6-os fejlesztő kártyán már láthattuk [8], hogy a választott FPGA chip TQFP tokozása miatt, két rétegű nyákon behuzalozható. Az FPGA NES kártya tervezésekor ez egy fő szempont volt, mivel így érdekes mérnöki megoldásokat kellett alkalmaznom a tág bekötése során, illetve a kártya elkészítési költségét is csökkenteni tudtam.

A prototípus kártyákat a jlpcb nevű kínai nyomtatott áramkör gyártó fogja gyártani. Ahhoz, hogy az Altium tervező program képes legyen bonyolultabb számítások (differential pair routing) és szimulációk készítésére, a réteg felépítést meg kell adnunk a PCB-nk számára. Ez a kínai gyártó által szolgáltatott információk által [2] alakítottam ki, ez az 5.5 ábrán látható.

| # | Name           | Material      | Type        | Weight | Thickness | Dk  |
|---|----------------|---------------|-------------|--------|-----------|-----|
|   | Top Overlay    |               | Overlay     |        |           |     |
|   | Top Solder     | Solder Resist | Solder Mask |        | 0.015mm   | 3.8 |
| 1 | Top Layer      |               | Signal      | 1oz    | 0.035mm   |     |
|   | Dielectric 1   | FR-4          | Dielectric  |        | 1.5mm     | 4.5 |
| 2 | Bottom Layer   |               | Signal      | 1oz    | 0.035mm   |     |
|   | Bottom Solder  | Solder Resist | Solder Mask |        | 0.015mm   | 3.8 |
|   | Bottom Overlay |               | Overlay     |        |           |     |

5.5. ábra. Az FPGA NES kártya réteg beállításai

### 5.11.2. Komponensek elhelyezése

A rétegbeállításokat követően, a komponensek és segéd áramkörei elhelyezése következett. Ennek segítségével tudtam meghatározni végül, hogy az FPGA mely I/O bankjaihoz/lába-ihoz fogom vezetni a különböző komponensek kivezetéseit, illetve ez határozta meg pcb-m méreteit is. Az alkatrészek és segéd áramkörei rajzolatát az F.1 függelékben láthatjuk, de a könnyebb áttekinthetőség érdekeben elkészítettem a következő ábrát:



5.6. ábra. A top layer alkatrész elhelyezési terve

FPGA NES kártya főbb alkotó komponensei:

1. Xilinx XC6SLX9-2TQG144C típusú FPGA
2. 256k x 16 bites (512 kB), 10 ns-os aszinkron SRAM (Cypress CY7C1041DV33-10ZSXI)
3. 3.5mm Jack csatlakozó (CUI SJ1-3553NG)
4. 100mW Erősítő (TS486IST fülhallgatókhöz)
5. Digital Analog Converter (DAC), (CS4334-KSZ)
6. HDMI csatlakozó
7. 2C szint illesztő (PCA9306DCUR)
8. Csatlakozó a LOGSYS fejlesztői kábel számára
9. 2.1mm 12 V / 5 A DC táp csatlakozó (DC10A)
10. 3,3 V feszültséget előállító tápegység
11. 1,2 V feszültséget előállító tápegység
12. 50 MHz-es oszcillátor
13. 7 pines NES GamePAD kontroller csatlakozók anya aljzat 1
14. 7 pines NES GamePAD kontroller csatlakozók anya aljzat 2
15. 32 Mbites SPI buszos soros FLASH (Atmel AT25DF321A)
16. MicroSD kártya foglalat
17. Az FPGA újrakonfigurálását indító gomb (PROG)
18. Az FPGA sikeres felkonfigurálását jelző LED (DONE)
19. Az FPGA kézi reset gomja (RSTn)
20. A bekapcsolt tápfeszültséget jelző LED (PWR)
21. A NES hangerő szabályzó gombjai (VOL-, VOL+)

Mivel kézi forrasztással és hőfűvő segítségével fogjuk a kártyát összeállítani, ezért törekedtem arra, hogy az összes komponens (és ezek segéd áramkörei) a előlap (top) oldalon helyezkedjen el, illetve a hátoldalra (bottom) csak azonos méretű elemek kerüljenek. Egyedül a HDMI csatlakozó biztosítéka került a hátoldalra, ami nem 0603-as méretű lett.

Ezt követően a PCB alkatrészek bekötésével foglalkoztam, itt a alsó és felső réteget is jel/föld (signal/GND) rétegként alakítottam ki. Ez azt jelenti, hogy jel vezetékeket és táp vonalak bekötését követően az egész nyák felszínén föld kitöltést hoztam létre, így növelte a jelek integritását. A alsó és a felső föld rétegeket, pedig via-k segítségével kötöttem össze (via stiching). A két réteget egyszerre az F.2 függelékben láthatjuk, az itt használt réteg ábrázolási mód az Altium designer átlátszó 2D módja, amely betekintést enged a föld kitöltések alá. A felső réteg jeleit és komponensek pad-jei vörös színnel vannak jelölve, a alsó réteg pedig kék színnel.

### 5.11.3. HDMI adatvonalainak bekötése

A HDMI vonalak kialakításáról már az 5.5 fejezet során olvashattunk. Viszont mivel a réteg kialakítás során a két rétegű megvalósítást választottunk és a PCB gyártó (jlcpcb) nem biztosít két réteg esetén impedancia kontrollálásra réteg felépítést. Ezért a HDMI szabványban leírt  $100 \Omega$ -os impedancia különbséget (15% os toleranciával) csak egy speciális differenciális pár bekötés típusával tudjuk biztosítani. Ez a típus a Dual Strip Coplanar Waweguide Grounded, ez azt jelenti, hogy differenciális pár bekötés mellett figyelnünk kell arra is, hogy a pár jobb és bal oldalán fix távolságra föld kitöltést helyezzünk el. Az itt található két réteg földjét előre meghatározott távolságokon via fence-szel látjuk el (ezt az elrendezést az 5.7 ábrán láthatjuk). A via-k közti maximális távolságot a következő képlettel számolhatjuk ki:

$$S(via) = \frac{\lambda}{20} = \frac{c}{20 * f * \sqrt{\epsilon_r}} \quad (5.4)$$

Az 5.4 képletben található  $\lambda$  a differenciális jelünk hullámhossza (a részletesebb képletben pedig:  $c$  a fény terjedési sebessége,  $f$  a jelünk frekvenciája,  $\epsilon_r$  pedig az anyag dielektromos állandója). A mi esetünkben ezen a vezeték páron 250 Mhz-es jeleket fogunk küldeni, ennek hullámhossza körülbelül 1,151 m, ebből kiszámított két via közti maximális távolság 0.058 m (ennél természetesen választhatunk kisebb értéket is).



**5.7. ábra.** Dual Strip Coplanar Waweguide Grounded felépítése

Az Altium designer segítségével van lehetőségem a GND réteg és vezető pár közti távolság kiszámítására (később az ecad szoftver ez alapján fogja elhelyezni a vezetékeket). Ezek a beállítások az alábbi ábrán láthatók:



**5.8. ábra.** Coplanar differential pair routings Altium beállításai

A FPGA NES kártya HDMI jeleinek bekötésére 2,75 mm-es via fence méretet választottam (körülbelül a fentebb kiszámolt érték 20-ad része) ez részletesen az 5.9 ábrán látható, amely az F.2 függelékből lett kiemelve.



**5.9. ábra.** A HDMI adatvonalaiknak bekötése

#### 5.11.4. FPGA táp vonalak kialakítása

Ahhoz, hogy a Spartan-6-os FPGA chip-et két rétegen teljes mértékben ki tudjuk használni egy speciális tápellátási módszerhez kellet folyamodnunk (Logsys-es fejlesztő kártya táp vonalai is hasonlóan vannak kialakítva). Ennek alapja az volt, hogy az alsó oldalról érkezik a 3.3 V és az 1.2 V-is. A 3.3 V-ot az FPGA lábai alatt végig vezetjük (innen indul ki a többi 3.3 V-os komponens tápellátást is) egy helyen be engedve az 1.2 V-ot a belső magja számára. Ezzel a rendszerrel az előnye, hogy így az összes csatoló (coupling) és hidegítő (bulk) kapacitás az FPGA alatt kaphat helyet, ezzel szabadon tartva a top réteget az FPGA I/O bankjainak és JTAG interfészének bekötésére.



**5.10. ábra.** FPGA tápellátásának kialakítása

A chip tápellátását az 5.10 ábrán láthatjuk, a kártya teljes tápellátását pedig az F.2 függelékben figyelhetjük meg.

### 5.11.5. FPGA NES 3D terve

A kártya tervezését követően, az Altium designer lehetőséget nyújt PCB 3D tervének megtekintéséhez (ez a komponensek elhelyezésénél, illetve nyák foglalatok/tokok tervezésénél is hasznos). Ez egy teljes képet nyújt a nyomtatott áramkör kialakításáról és jövőbeli kinézetéről. Az FPGA NES 3D tervét a következő (5.11) ábrán láthatjuk:



5.11. ábra. 3D PCB rajzolat

## 6. fejezet

# FPGA tervezés

A Nintendo Entertainment System hardveres komponenseinek újratervezése során az ISE Design Suit szoftvert és hardver leíró nyelvnek pedig a Verilog-ot választottam. A tervezés fő szempontja a beláthatóság, céltudatosság és tesztelhetőség volt, ez elengedhetetlen egy nagyobb projekt fejlesztése során. A megtervezett chipeknek az összehangolt és hibátlan (az eredeti NES hardverrel megegyező) működését követően léptem tovább a következő fejlesztési egységre.

A következőkben a diplomatervezés során tervezett és tesztelt hardveres komponensek működését és felépítést fogom bemutatni, az úgynevezett "top-down" elv alapján. Ennek jelentése, hogy először a teljes rendszer kapcsolatát mutatom be, majd ezt követően térek ki a különböző elemek részletes működésére.

### 6.1. Rendszer blokkvázlat bemutatása

Az FPGA tervezés során a különböző funkciókat ellátó hardveres elemek (a NES különböző chipjeinek) leírását modulokra osztottam. Ezeket a modulokat és a köztük lévő logikai kapcsolatot láthatjuk a 6.1 ábrán.

Az ábrán a modulok közti két fő adatkapcsolatot láthatjuk



6.1. ábra. Megvalósult NES top rendszer diagramja

### 6.1.1. Adatbuszok implementálása

A NES eredeti belső adatbuszkapcsolatait alapul véve, két fő adatbuszt különböztetünk meg a CPU és a PPU adatbuszt. A PPU adatbusz struktúrájában egyszerűbb csupán a memória menedzser modullal kommunikál. A modulban található név tábla (NT) memóriát írja és olvassa, illetve a karakter ROM tartalmát olvassa. A CPU adatbusz működése a DMA miatt egy fokkal bonyolultabb. Alap esetben a CPU adatbuszán keresztül éri el a címtartományában található különböző perifériák regisztereit. A slave interfések implementálása során ügyeltem arra hogy a különböző hardveres komponensek minél egyszerűbb interfésszel fogadják a CPU adatbuszát (ezzel egyszerűsítve a szintetizálálandó hardveren).

```

1 // CPU data in
2 wire [7:0] cpu_din = (memory_manager_cpu_din | ppu_to_cpu_din | controller_cpu_din);
3
4 // DMA and CPU outputs
5 assign cpu_addr = (dma_cpu_valid) ? (dma_cpu_addr) : (ag6502_addr);
6 assign cpu_dout = (dma_cpu_dout | ag6502_dout);
7 assign rnw = ~(~dma_cpu_rnw | ~ag6502_rnw);

```

### 6.1. hardverleírás részlet. CPU adatbuszt befolyásoló DMA

```

1 // PPU Slave interface bemenet
2 .slv_mem_addr(cpu_addr[2:0]), // register interface reg select (#2000-#2007)
3 .slv_mem_cs((cpu_addr[15:13] == 3'b001)), // register interface enable (#2000 - #3FFF just
when it is active)
4
5 // Controller inputs
6 .cpu_data_in(cpu_dout[0]), //just one bit the $4016 write strobe
7
8 // DMA Slave databus
9 .slv_mem_select((ag6502_addr[15:14] == 2'b01)),

```

```
10 .slv_mem_addr(ag6502_addr[4:0]),
```

### 6.1.2. Működési órajel választása

```
1 reg clkgen_cnt_en_clr;
2 wire clkgen_cnt_en_set = (x_rendercntr == (ODDFRAME_END_OF_BG_RENDERING_LINE + 4));
3
4 always @(*)
5 begin
6     if (background_enabled && oddframe && (y_renderingcntr == PRERENDERING_ROW))
7         clkgen_cnt_en_clr <= (x_rendercntr == ODDFRAME_END_OF_BG_RENDERING_LINE);
8     else
9         clkgen_cnt_en_clr <= 1'b0;
10 end
11
12 // clock generation enable
13 reg clkgen_cnt_en;
14
15 always @ (posedge clk)
16 begin
17     if (rst || (x_rendercntr == END_OF_BG_RENDERING_LINE) || clkgen_cnt_en_clr)
18         clkgen_cnt_en <= 1'b0;
19     else
20         if ((x_rendercntr == FIRST_SCANLINE_PIXEL) || clkgen_cnt_en_set)
21             clkgen_cnt_en <= 1'b1;
22 end
23
24 //clock generation timer
25 reg [3:0] clkgen_cnt;
26
27 always @ (posedge clk)
28 begin
29     if (rst)
30         clkgen_cnt <= 4'd0;
31     else
32         if (clkgen_cnt_en)
33             if (clkgen_cnt == 4'd11)
34                 clkgen_cnt <= 4'd0;
35             else
36                 clkgen_cnt <= clkgen_cnt + 4'd1;
37 end
38
39 always @ (posedge clk)
40 begin
41     ph1_rising <= clkgen_cnt_en & (clkgen_cnt == 4'd0);
42     ph1_falling <= clkgen_cnt_en & (clkgen_cnt == 4'd5);
43     ph2_rising <= clkgen_cnt_en & (clkgen_cnt == 4'd6);
44     ph2_falling <= clkgen_cnt_en & (clkgen_cnt == 4'd11);
45 end
```

## 6.2. Picture Process Unit

### 6.2.1. Eredeti rendrelés menete

```
1 reg [7:0] bg_lsb_buff_reg;
2
3 always @ (posedge clk)
4 begin
5     if (rst)
6         bg_lsb_buff_reg <= 8'd0;
7     else
8         if (bg_msb_read)
9             bg_lsb_buff_reg <= bg_lsb_reg; // shift reg
10        else if ((x_rendercntr[1:0] == 2'b11) && (x_rendercntr > START_OF_SHIFT) && (
11            x_rendercntr <= END_OF_SHIFT) && ~(bgrender_state == VBLANK))
12            bg_lsb_buff_reg <= {bg_lsb_buff_reg[6:0], 1'b0};
13 end
```

```

13    reg [7:0] shr_lsb_render = 8'd0;
14    wire      bg_lsb_out;
15
16    always @(posedge clk)
17        if ((x_rendercntr[1:0] == 2'b11) && (x_rendercntr > START_OF_SHIFT) && (x_rendercntr <=
18            END_OF_SHIFT) && ~ (bgrender_state == VBLANK))
19            shr_lsb_render <= {shr_lsb_render[6:0], bg_lsb_buff_reg[7]};
20
21    assign bg_lsb_out = shr_lsb_render[~fh_reg];
22
23    reg [7:0] shr_msb_render = 8'd0;
24    wire      bg_msb_out;
25
26    always @(posedge clk)
27        if ((x_rendercntr[1:0] == 2'b11) && (x_rendercntr > START_OF_SHIFT) && (x_rendercntr <=
28            END_OF_SHIFT) && ~ (bgrender_state == VBLANK))
29            shr_msb_render <= {shr_msb_render[6:0], bg_msb_reg[7]};
30
31    assign bg_msb_out = shr_msb_render[~fh_reg];
32
33    localparam H_SPRITE0_CHECK_END = 4*287 - 1;
34
35    reg sprite0_check_reg;
36
37    always @(posedge clk)
38    begin
39        if (rst || (x_rendercntr == H_SPRITE0_CHECK_END))
40            sprite0_check_reg <= 0;
41        else
42            if (x_rendercntr == FIRST_SCANLINE_PIXEL)
43                sprite0_check_reg <= 1;
44    end
45
46
47    wire sprite0_check = sprite0_check_reg & next_pixel;
48
49    wire visible_bg_pixel = |bg_pixel[1:0];
50
51    reg [1:0] tile_attr_reg_saved;
52
53    always @(posedge clk)
54    begin
55        if (rst)
56            tile_attr_reg_saved <= 2'b0;
57        else
58            if (bg_msb_read)
59                tile_attr_reg_saved <= tile_attr_reg;
60    end
61
62
63    reg [7:0] shr_attr0_render = 8'd0;
64
65    always @(posedge clk)
66        if ((x_rendercntr[1:0] == 2'b11) && (x_rendercntr > START_OF_SHIFT) && (x_rendercntr <=
67            END_OF_SHIFT) && ~ (bgrender_state == VBLANK))
68            shr_attr0_render <= {shr_attr0_render[6:0], tile_attr_reg_saved[0]};
69
70    wire shr_attr0_out = shr_attr0_render[~fh_reg];
71
72    reg [7:0] shr_attr1_render = 8'd0;
73
74    always @(posedge clk)
75        if ((x_rendercntr[1:0] == 2'b11) && (x_rendercntr > START_OF_SHIFT) && (x_rendercntr <=
76            END_OF_SHIFT) && ~ (bgrender_state == VBLANK))
77            shr_attr1_render <= {shr_attr1_render[6:0], tile_attr_reg_saved[1]};
78
79    wire shr_attr1_out = shr_attr1_render[~fh_reg];
80
81    reg [3:0] bg_pixel;
82
83    always @(posedge clk)
84    begin
85        if (rst)
86            bg_pixel <= 4'b0;
87        else
88            if (bg_msb_out)
89                bg_pixel <= bg_msb_out;
90            else
91                bg_pixel <= bg_lsb_out;
92    end
93
```

```

50         bg_pixel <= 4'b0;
51     else
52         bg_pixel <= {shr_attr1_out, shr_attr0_out, bg_msb_out, bg_lsb_out};
53     end
54
55     wire [4:0] palette_addr =  (sprite_priority & visible_bg_pixel) ? ({1'b0, bg_pixel}) : ({1'b1,
56     sprite_pixel});
57
58     wire tarnsparent_bground = ({bg_msb_out, bg_lsb_out} == 2'b00);
59     wire tarnsparent_sprite = (sprite_pixel == 2'b00);
60
61     wire sprite_select = (~sprite_priority | tarnsparent_bground) & ~tarnsparent_sprite;
62
63     assign sprite0_hit_set = visible_bg_pixel & sprite0_visible & sprite0_check;

```

### 6.2.2. VGA rendelés

### 6.2.3. Háttér renderelési állapot gép

#### 6.2.4. Sprite rendering állapot gép

#### 6.2.5. CPU által elérhető regiszterek és CPU adatbusz

```

1 //register write enable signals
2 wire control_wr      = ph2_falling & slv_mem_cs & ~slv_mem_rnw & (slv_mem_addr == 3'b000); //CTRL
3   register write #2000
4 wire render_mask_wr = ph2_falling & slv_mem_cs & ~slv_mem_rnw & (slv_mem_addr == 3'b001); //MASK
5   register write #2001
6 wire oam_addr_wr    = ph2_falling & slv_mem_cs & ~slv_mem_rnw & (slv_mem_addr == 3'b011); //OAM
7   read/write address #2003
8 wire oam_data_wr    = ph2_falling & slv_mem_cs & ~slv_mem_rnw & (slv_mem_addr == 3'b100); //OAM
9   data read/write #2004
10
11 wire scrolling_wr   = ph2_falling & slv_mem_cs & ~slv_mem_rnw & (slv_mem_addr == 3'b101); //fine
12   scroll position (two writes: X scroll, Y scroll) #2005
13 wire vram_addr_wr   = ph2_falling & slv_mem_cs & ~slv_mem_rnw & (slv_mem_addr == 3'b110); //PPU
14   read/write address (two writes: most significant byte, least significant byte) #2006
15 wire vram_data_wr   = ph2_falling & slv_mem_cs & ~slv_mem_rnw & (slv_mem_addr == 3'b111); //PPU
16   data read/write #2007
17
18 //Register read enable signals
19 wire status_rd       = slv_mem_cs & slv_mem_rnw & (slv_mem_addr == 3'b010);
20 wire oam_data_rd     = slv_mem_cs & slv_mem_rnw & (slv_mem_addr == 3'b100);
21 wire vram_data_rd    = slv_mem_cs & slv_mem_rnw & (slv_mem_addr == 3'b111);

```

#### 6.2.6. PPU adatbusz és memória elérése

### 6.3. NES memória felépítése FPGA-ban

#### 6.4. DMA

## 6.5. 6502 processzor működése

## 6.6. NES kontrollerek kezelése

## **7. fejezet**

# **A NES tesztelése**

### **7.1. Rendszer szimulációk**

### **7.2. Hardveres tesztek**

#### **7.2.1. Donkey Kong**

#### **7.2.2. Super Mario Bros.**

### **7.3. A tesztek eredményeinek kiértékelése**

## 8. fejezet

# Összefoglalás, jövőbeli tervezek

A Diplomatervezés 1 során sikerült foglalkoznom a projekt irodalom kutatásával, és NES nyomtatott áramkörének FPGA alapú újratervezésével, illetve a PPU hardverének felépítésével és felújításával.

A Diplomatervezés végeztéig még meg kell ismernem a CPU és APU hardverét is részletesen, illetve ezeket a PPU-val együtt meg kell terveznem a spatan-6-os FPGA-ba. Ezt követően pedig tesztelni kell ezeket az éles hardveren.

A projekt hosszútávú tervei közé tartozik az MicroSD kártyáról történő játék betöltés és minél több NES játék mapper-ének lefejlesztése, ezzel teljes körű hardveres emulálást készítve a Nintendo Entertainment System-hez.

# Köszönetnyilvánítás

Ez nem kötelező, akár törölhető is. Ha a szerző szükségét érzi, itt lehet köszönetet nyilvánítani azoknak, akik hozzájárultak munkájukkal ahhoz, hogy a hallgató a szakdolgozatban vagy diplomamunkában leírt feladatokat sikeresen elvégezze. A konzulensnek való köszönetnyilvánítás sem kötelező, a konzulensnek hivatalosan is dolga, hogy a hallgatót konzultálja.

# Ábrák jegyzéke

|       |                                                                               |    |
|-------|-------------------------------------------------------------------------------|----|
| 3.1.  | Nintendo Entertainment System . . . . .                                       | 3  |
| 3.2.  | Nintendo Entertainment System NTSC alaplap [5] . . . . .                      | 4  |
| 3.3.  | Nintendo Entertainment System játék kazetta [4] . . . . .                     | 5  |
| 3.4.  | Komponensek közti kapcsolat (fent játék kazetta) [5] . . . . .                | 6  |
| 3.5.  | Katódsugárcsöves TV-k működése . . . . .                                      | 8  |
| 3.6.  | Super Mario Bros. Pattern táblái . . . . .                                    | 9  |
| 3.7.  | Gumba (Mario egyik ellenfele) bal felső csempe elem szín adatai [?] . . . . . | 9  |
| 3.8.  | A MOS 6502 egyedi nyolcbites architektúrája . . . . .                         | 13 |
| 3.9.  | A MOS 6502 processzor státusz regisztere (P) . . . . .                        | 14 |
| 3.10. | A 2A03-ban található 6502 teljes utasítás készlete . . . . .                  | 15 |
| 3.11. | Az audió feldolgozó egység hang sávjai . . . . .                              | 15 |
| 4.1.  | A NES játék konzol dimenziói . . . . .                                        | 20 |
| 5.1.  | NES kártya blokkdiagramja . . . . .                                           | 21 |
| 5.2.  | NES audió jelért felelős áramkörök . . . . .                                  | 24 |
| 5.3.  | aljzat anya . . . . .                                                         | 25 |
| 5.4.  | NES kontroller csatlakozó . . . . .                                           | 26 |
| 5.5.  | Az FPGA NES kártya réteg beállításai . . . . .                                | 28 |
| 5.6.  | A top layer alkatrész elhelyezési terve . . . . .                             | 28 |
| 5.7.  | Dual Strip Coplanar Waweguide Grounded felépítése . . . . .                   | 30 |
| 5.8.  | Coplanar differential pair routings Altium beállításai . . . . .              | 30 |
| 5.9.  | A HDMI adatvonalainak bekötése . . . . .                                      | 31 |
| 5.10. | FPGA tápellátásának kialakítása . . . . .                                     | 31 |
| 5.11. | 3D PCB rajzolat . . . . .                                                     | 32 |
| 6.1.  | Megvalósult NES top rendszer diagramja . . . . .                              | 34 |

# Táblázatok jegyzéke

|                                                                                 |    |
|---------------------------------------------------------------------------------|----|
| 3.1. A PPU memória kezelése (14 bit címek) mirroring jelenséggel . . . . .      | 10 |
| 3.2. A CPU memória címtartománya (16 bit címek) mirroring jelenséggel . . . . . | 15 |
| 5.1. HDMI lábkiosztás . . . . .                                                 | 25 |
| 5.2. Fejlesztői port bekötése . . . . .                                         | 27 |

# Irodalomjegyzék

- [1] Nathan Altice: *I Am Error - The Nintendo Family Computer / Entertainment System Platform*. September 8, 2017, The MIT Press.
- [2] JLPCB: *PCB Manufacturing and Assembly Capabilities*. JLPCB, 2023. 06. <https://jlcpcb.com/capabilities/pcb-capabilities>.
- [3] NESDev wiki community: Nes reference guide (2023. november 26.). [https://www.nesdev.org/wiki/NES\\_reference\\_guide](https://www.nesdev.org/wiki/NES_reference_guide).
- [4] NesHacker youtube csatorna: Nes carts explained (2023. október 21.). <https://www.youtube.com/watch?v=GssRNEaKoPw>.
- [5] NesHacker youtube csatorna: Nes hardware explained (2023. október 21.). <https://www.youtube.com/watch?v=mMq4FFUnBPc>.
- [6] NesHacker youtube csatorna: Neshacker (2023. november 26.). <https://www.youtube.com/@NesHacker>.
- [7] Cypress Perform: *CY7C1041DV33 - 4-Mbit (256 K × 16) Static RAM*. Cypress Perform, 2023. 06. [https://datasheet.lcsc.com/lcsc/1804240720\\_Cypress-Semicon-CY7C1041DV33-10ZSXI\\_C32338.pdf](https://datasheet.lcsc.com/lcsc/1804240720_Cypress-Semicon-CY7C1041DV33-10ZSXI_C32338.pdf).
- [8] Raikovich Tamás: *LOGSYS SPARTAN-6 FPGA KÁRTYA (V2.1) FELHASZNÁLÓI ÚTMUTATÓ*. Budapesti Műszaki és Gazdaságtudományi Egyetem, 2023. 06. [http://logsys.mit.bme.hu/sites/default/files/page/2009/09/LOGSYS\\_SP6\\_FPGA\\_Board.pdf](http://logsys.mit.bme.hu/sites/default/files/page/2009/09/LOGSYS_SP6_FPGA_Board.pdf).

# Függelék

## F.1. NES kártya alkatrész elhelyezési terve

### F.1.1. Top



### F.1.2. Bottom



#### F.2. Nyomtatott áramköri terve (2D transparent)



## F.3. FPGA NES kártya kapcsolási rajza

### F.3.1. Tápegység



### F.3.2. HDMI és MicroSD kártya csatlakozó



### F.3.3. DAC, erősítő és kontroller áramkörök



### F.3.4. SRAM és SPI-Flash



### F.3.5. FPGA OSC és JTAG



### F.3.6. FPGA IO bankok

