

# Game Boy w języku C#

## Podstawy emulacji niskopoziomowej

**Szymon Sandura**  
.NET Developer / Tech Lead

# Zagadnienia

- Czym jest emulator?
- Emulator konsoli Nintendo Gameboy
  - Architektura
  - Przykłady implementacji
- Pokaz emulatora

# Czym jest emulator?

**Emulator** to program lub  sprzęt, który imituje działanie innego programu lub sprzętu.

Pojęcie jest dość otwarte i często się przenika z takimi pojęciami jak

- *Symulator*
- *Maszyna wirtualna*
- *Port*
- *Adapter*
- *Warstwa kompatybilności (ang. compatibility layer)*

# WINE - Wine Is Not Emulator





ニンテンドークラシックミニ  
**FAMILY COMPUTER**  
ファミリーコンピュータ

**Nintendo**  
ENTERTAINMENT SYSTEM™

ニンテンドー64  
**SUPER Famicom**  
スーパーファミコン

**GAME BOY**

**SUPER NINTENDO**  
ENTERTAINMENT SYSTEM

**NINTENDO 64**

**GAME BOY ADVANCE**

 NINTENDO  
**GAMECUBE**

**GAME BOY**  
PLAYER

 e-reader

NINTENDO  DS Lite

**Wii**

:new:  
NINTENDO  3DS XL

**Wii U**

 NINTENDO  
**SWITCH**

PlayStation. 

PocketStation™

  
PlayStation.2

  
PlayStation.3

  
PlayStation. Portable

  
PlayStation. Portable

  
PlayStation.4

  
PlayStation. VITA.

  
PlayStation.5

**SEGA**

  
Dreamcast.

  
JAGUAR  
64-BIT

  
SNK  
**NEOGEO**

  
XBOX 360

  
XBOX ONE

  
SERIES  
**X|S**

  
GAME PASS



# Korzyści emulatorów konsol

- Zachowanie historii gier
- Możliwość zrozumienia gier i sprzętu
- **Usprawnienia graficzne** (płynność, rozdzielczość, wybór ekranu, mody graficzne)
- Możliwość wykorzystania **dowolnego kontrolera** (np. klawiatury), **save state'ów**, **online**

# Emulacja generalnie jest legalna ale nie wolno...

- Udostępniać kopii gier (np. pliki ROM, ISO)
- Udostępniać BIOSu konsoli, kluczy deszyfrujących itp.
- Naruszać patentów, znaków towarowych
- Przyciągać uwagi Nintendo :)

# Techniki emulacji?

- **LLE (Low-Level Emulation)**
- **HLE (High-Level Emulation)**

Obsługa CPU:

- **Interpreter**
- Kompilator (JIT, AOT)

# LLE - Emulacja niskopoziomowa

- Szczegółowa implementacja architektury i komunikacji między komponentami
- Wysoka precyzja odwzorowania
- Spory narzut wydajności...

# Dlaczego emulacja jest kosztowna?

- Procesor Gameboya 4Mhz  $\approx$  4 miliony cykli na sekundę, do **milionów instrukcji na sekundę**
- Nasz procesor musi wykonać X instrukcji na każdą instrukcję, bo...
  - Nie piszemy w Assemblerze
  - Różnice w architekturze

# HLE - High Level Emulation

- Skupia się na efekcie końcowym, a nie na detalach architektury i precyzji
- GPU, Dźwięk, BIOS, ale **nie CPU!**
- Pomość między natywnymi API
  - Przykład: Playstation zleciło swojemu API narysowanie trójkąta w 3D
  - Wywołujemy najbardziej zblizoną metodę w OpenGL/Vulkan/DirectX, która rysuje trójkąt
- Potencjalnie prędkość bliska natywnej

# Nintendo Game Boy

- Sprzedawany w latach 1989 - 2003  
**(118,93 mln sztuk)**
- **CPU** Sharp LR35902 **4,19 Mhz**
- Dedykowany układ **Pixel Processing Unit (PPU, prymitywne GPU)**, **8KB VRAM**
- Ekran LCD 2,5 cala **160x144px ~59,7Hz**
- Port szeregowy (**Multiplayer...i drukarka?**)
- **4 kanały** audio z wyjściem mono/stereo



# Game Boy Printer (1998)



# Game Boy Camera (1998)

- 0,1 Mpx!



# CPU

- **4,19 Mhz**  $\approx$  4 miliony cykli na sekundę
- Operacje zajmują od 4-20 cykli
- **8-bitowy**
- Wspiera adresację pamięci do 64KB (indeksy od **0** do **65536**)
- Wspiera **przerwania** (ang. **Interrupts**)
- Zawiera liczne **rejestry**



# CPU - Rejestry

- **8** rejestrów 8-bitowych...  
A, F, B, C, D, E, H, L
- ...albo 4 rejesty 16-bitowe  
AF, BC, DE, HL
- SP - Stack Pointer, adres ostatniego elementu stosu (16-bit)
- **PC - Program Counter**, adres najbliższej instrukcji do wykonania (16-bit)

| Registers |   |
|-----------|---|
| A         | F |
| B         | C |
| D         | E |
| H         | L |
| SP        |   |
| PC        |   |

## Structure

  ▼  CpuRegister

-  R8LookupTable : byte[]
-  CpuRegister()
-  InterruptsMasterEnabled : bool
-  IMEPending : bool
-  SP : ushort
-  PC : ushort
-  AF : ushort
-  A : byte
-  F : byte
-  ZeroFlag : bool
-  NegativeFlag : bool
-  HalfCarryFlag : bool
-  CarryFlag : bool
-  BC : ushort
-  B : byte
-  C : byte
-  DE : ushort
-  D : byte
-  E : byte
-  HL : ushort
-  H : byte
-  L : byte





```
Set8BitIncrementCarryFlags(value);
Register.SetRegisterByR8(r8, value.Add(1));
return (instructionBytesLength: 1, durationTStates: 4);
```

```
private void Set8BitIncrementCarryFlags(byte a)
{
    Register.NegativeFlag = false;
    Register.ZeroFlag = (byte)(a + 1) == 0;
    Register.HalfCarryFlag = (a & 0xF) + 1 > 0xF;
}
```

**Eric's Blog**

# **WTF is the DAA instruction?**

January 30, 2018

Źródło: <https://ehaskins.com/2018-01-30%20Z80%20DAA/>



**Ille takich metod musimy zaprogramować?**

# 16 x 16 = 256...

|    | x0                 | x1                | x2                   | x3               | x4                     | x5               | x6                 | x7                  | x8                  | x9                | xA                  | xB               | xC                    | xD               | xE                | xF              |
|----|--------------------|-------------------|----------------------|------------------|------------------------|------------------|--------------------|---------------------|---------------------|-------------------|---------------------|------------------|-----------------------|------------------|-------------------|-----------------|
| 0x | NOP<br>1 4         | LD BC,d16<br>3 12 | LD (BC),A<br>1 8     | INC BC<br>1 8    | INC B<br>1 4           | DEC B<br>2 8     | RLCA<br>1 4        | LD (a16),SP<br>3 20 | ADD HL,BC<br>1 8    | LD A,(BC)<br>1 8  | DEC BC<br>1 8       | INC C<br>1 4     | DEC C<br>1 4          | LD C,d8<br>2 8   | RRCA<br>1 4       |                 |
| 1x | STOP 0<br>2 4      | LD DE,d16<br>3 12 | LD (DE),A<br>1 8     | INC DE<br>1 8    | INC D<br>1 4           | DEC D<br>2 8     | RLA<br>1 4         | JR r8<br>2 12       | ADD HL,DE<br>1 8    | LD A,(DE)<br>1 8  | DEC DE<br>1 8       | INC E<br>1 4     | DEC E<br>1 4          | LD E,d8<br>2 8   | RRA<br>1 4        |                 |
| 2x | JR NZ,r8<br>2 12/8 | LD HL,d16<br>3 12 | LD (HL),A<br>1 8     | INC HL<br>1 8    | INC H<br>1 4           | DEC H<br>2 8     | LD H,d8<br>1 4     | DAA<br>2 12/8       | ADD HL,HL<br>1 8    | LD A,(HL+)<br>1 8 | DEC HL<br>1 8       | INC L<br>1 4     | DEC L<br>1 4          | LD L,d8<br>2 8   | CPL<br>1 4        |                 |
| 3x | JR NC,r8<br>2 12/8 | LD SP,d16<br>3 12 | LD (HL-),A<br>1 8    | INC SP<br>1 8    | INC (HL)<br>1 12       | DEC (HL)<br>1 12 | LD (HL),d8<br>1 22 | SCF<br>1 4          | JR C,r8<br>2 12/8   | ADD HL,SP<br>1 8  | LD A,(HL-)<br>1 8   | DEC SP<br>1 8    | INC A<br>1 4          | DEC A<br>1 4     | LD A,d8<br>2 8    | CCF<br>1 4      |
| 4x | LD B,B<br>1 4      | LD B,C<br>1 4     | LD B,D<br>1 4        | LD B,E<br>1 4    | LD B,H<br>1 4          | LD B,L<br>1 4    | LD B,(HL)<br>1 8   | LD B,A<br>1 4       | LD C,B<br>1 4       | LD C,C<br>1 4     | LD C,D<br>1 4       | LD C,E<br>1 4    | LD C,H<br>1 4         | LD C,L<br>1 8    | LD C,(HL)<br>1 4  | LD C,A          |
| 5x | LD D,B<br>1 4      | LD D,C<br>1 4     | LD D,D<br>1 4        | LD D,E<br>1 4    | LD D,H<br>1 4          | LD D,L<br>1 4    | LD D,(HL)<br>1 8   | LD D,A<br>1 4       | LD E,B<br>1 4       | LD E,C<br>1 4     | LD E,D<br>1 4       | LD E,E<br>1 4    | LD E,H<br>1 4         | LD E,L<br>1 8    | LD E,(HL)<br>1 4  | LD E,A          |
| 6x | LD H,B<br>1 4      | LD H,C<br>1 4     | LD H,D<br>1 4        | LD H,E<br>1 4    | LD H,H<br>1 4          | LD H,L<br>1 4    | LD H,(HL)<br>1 8   | LD H,A<br>1 4       | LD L,B<br>1 4       | LD L,C<br>1 4     | LD L,D<br>1 4       | LD L,E<br>1 4    | LD L,H<br>1 4         | LD L,L<br>1 8    | LD L,(HL)<br>1 4  | LD L,A          |
| 7x | LD (HL),B<br>1 8   | LD (HL),C<br>1 8  | LD (HL),D<br>1 8     | LD (HL),E<br>1 8 | LD (HL),H<br>1 8       | HALT<br>1 4      | LD (HL),A<br>1 8   | LD A,B<br>1 4       | LD A,C<br>1 4       | LD A,D<br>1 4     | LD A,E<br>1 4       | LD A,H<br>1 4    | LD A,L<br>1 8         | LD A,(HL)<br>1 8 | LD A,A            |                 |
| 8x | ADD A,B<br>1 4     | ADD A,C<br>1 4    | ADD A,D<br>1 4       | ADD A,E<br>1 4   | ADD A,H<br>1 4         | ADD A,L<br>1 4   | ADD A,(HL)<br>1 8  | ADD A,A<br>1 4      | ADC A,B<br>1 4      | ADC A,C<br>1 4    | ADC A,D<br>1 4      | ADC A,E<br>1 4   | ADC A,H<br>1 4        | ADC A,L<br>1 8   | ADC A,(HL)<br>1 4 | ADC A,A         |
| 9x | SUB B<br>1 4       | SUB C<br>1 4      | SUB D<br>1 4         | SUB E<br>1 4     | SUB H<br>1 4           | SUB L<br>1 4     | SUB (HL)<br>1 8    | SUB A<br>1 4        | SBC A,B<br>1 4      | SBC A,C<br>1 4    | SBC A,D<br>1 4      | SBC A,E<br>1 4   | SBC A,H<br>1 4        | SBC A,L<br>1 8   | SBC A,(HL)<br>1 4 | SBC A,A         |
| Ax | AND B<br>1 4       | AND C<br>1 4      | AND D<br>1 4         | AND E<br>1 4     | AND H<br>1 4           | AND L<br>1 4     | AND (HL)<br>1 8    | AND A<br>1 4        | XOR B<br>1 4        | XOR C<br>1 4      | XOR D<br>1 4        | XOR E<br>1 4     | XOR H<br>1 4          | XOR L<br>1 8     | XOR (HL)<br>1 4   | XOR A           |
| Bx | OR B<br>1 4        | OR C<br>1 4       | OR D<br>1 4          | OR E<br>1 4      | OR H<br>1 4            | OR L<br>1 4      | OR (HL)<br>1 8     | OR A<br>1 4         | CP B<br>1 4         | CP C<br>1 4       | CP D<br>1 4         | CP E<br>1 4      | CP H<br>1 4           | CP L<br>1 8      | CP (HL)<br>1 4    | CP A            |
| Cx | RET NZ<br>1 20/8   | POP BC<br>1 12    | JP NZ,a16<br>3 16/12 | JP a16<br>3 16   | CALL NZ,a16<br>3 24/12 | PUSH BC<br>1 16  | ADD A,d8<br>2 8    | RST 00H<br>1 16     | RET Z<br>1 20/8     | RET<br>1 16       | JP Z,a16<br>3 16/12 | PREFIX CB<br>1 4 | CALL Z,a16<br>3 24/12 | CALL a16<br>3 24 | ADC A,d8<br>2 8   | RST 08H<br>1 16 |
| Dx | RET NC<br>1 20/8   | POP DE<br>1 12    | JP NC,a16<br>3 16/12 |                  | CALL NC,a16<br>3 24/12 | PUSH DE<br>1 16  | SUB d8<br>2 8      | RST 10H<br>1 16     | RET C<br>1 20/8     | RETI<br>1 16      | JP C,a16<br>3 16/12 |                  | CALL C,a16<br>3 24/12 |                  | SBC A,d8<br>2 8   | RST 18H<br>1 16 |
| Ex | LDH (a8),A<br>2 12 | POP HL<br>1 12    | LD (C),A<br>2 8      |                  |                        | PUSH HL<br>1 16  | AND d8<br>2 8      | RST 20H<br>1 16     | ADD SP,r8<br>2 16   | JP (HL)<br>1 4    | LD (a16),A<br>3 16  |                  |                       |                  | XOR d8<br>2 8     | RST 28H<br>1 16 |
| Fx | LDH A,(a8)<br>2 12 | POP AF<br>1 12    | LD A,(C)<br>2 8      | DI<br>1 4        |                        | PUSH AF<br>1 16  | OR d8<br>2 8       | RST 30H<br>1 16     | LD HL,SP+r8<br>2 12 | LD SP,HL<br>1 8   | LD A,(a16)<br>3 16  | EI<br>1 4        |                       |                  | CP d8<br>2 8      | RST 38H<br>1 16 |

...+256 = 512

|    | x0                        | x1                        | x2                        | x3                        | x4                        | x5                        | x6                            | x7                        | x8                        | x9                        | xA                        | xB                        | xC                        | xD                            | xE                           | xF                       |
|----|---------------------------|---------------------------|---------------------------|---------------------------|---------------------------|---------------------------|-------------------------------|---------------------------|---------------------------|---------------------------|---------------------------|---------------------------|---------------------------|-------------------------------|------------------------------|--------------------------|
| 0x | RLC B<br>2 8<br>Z 0 0 C   | RLC C<br>2 8<br>Z 0 0 C   | RLC D<br>2 8<br>Z 0 0 C   | RLC E<br>2 8<br>Z 0 0 C   | RLC H<br>2 8<br>Z 0 0 C   | RLC L<br>2 8<br>Z 0 0 C   | RLC (HL)<br>2 16<br>Z 0 0 C   | RLC A<br>2 8<br>Z 0 0 C   | RR C B<br>2 8<br>Z 0 0 C  | RR C<br>2 8<br>Z 0 0 C    | RR D<br>2 8<br>Z 0 0 C    | RR E<br>2 8<br>Z 0 0 C    | RR H<br>2 8<br>Z 0 0 C    | RR L<br>2 8<br>Z 0 0 C        | RR C (HL)<br>2 16<br>Z 0 0 C | RR C A<br>2 8<br>Z 0 0 C |
| 1x | RL B<br>2 8<br>Z 0 0 C    | RL C<br>2 8<br>Z 0 0 C    | RL D<br>2 8<br>Z 0 0 C    | RL E<br>2 8<br>Z 0 0 C    | RL H<br>2 8<br>Z 0 0 C    | RL L<br>2 8<br>Z 0 0 C    | RL (HL)<br>2 16<br>Z 0 0 C    | RL A<br>2 8<br>Z 0 0 C    | RR B<br>2 8<br>Z 0 0 C    | RR C<br>2 8<br>Z 0 0 C    | RR D<br>2 8<br>Z 0 0 C    | RR E<br>2 8<br>Z 0 0 C    | RR H<br>2 8<br>Z 0 0 C    | RR L<br>2 8<br>Z 0 0 C        | RR (HL)<br>2 16<br>Z 0 0 C   | RR A<br>2 8<br>Z 0 0 C   |
| 2x | SLA B<br>2 8<br>Z 0 0 C   | SLA C<br>2 8<br>Z 0 0 C   | SLA D<br>2 8<br>Z 0 0 C   | SLA E<br>2 8<br>Z 0 0 C   | SLA H<br>2 8<br>Z 0 0 C   | SLA L<br>2 8<br>Z 0 0 C   | SLA (HL)<br>2 16<br>Z 0 0 C   | SLA A<br>2 8<br>Z 0 0 C   | SRA B<br>2 8<br>Z 0 0 C   | SRA C<br>2 8<br>Z 0 0 C   | SRA D<br>2 8<br>Z 0 0 C   | SRA E<br>2 8<br>Z 0 0 C   | SRA H<br>2 8<br>Z 0 0 C   | SRA L<br>2 8<br>Z 0 0 C       | SRA (HL)<br>2 16<br>Z 0 0 C  | SRA A<br>2 8<br>Z 0 0 C  |
| 3x | SWAP B<br>2 8<br>Z 0 0 C  | SWAP C<br>2 8<br>Z 0 0 C  | SWAP D<br>2 8<br>Z 0 0 C  | SWAP E<br>2 8<br>Z 0 0 C  | SWAP H<br>2 8<br>Z 0 0 C  | SWAP L<br>2 8<br>Z 0 0 C  | SWAP (HL)<br>2 16<br>Z 0 0 C  | SWAP A<br>2 8<br>Z 0 0 C  | SRL B<br>2 8<br>Z 0 0 C   | SRL C<br>2 8<br>Z 0 0 C   | SRL D<br>2 8<br>Z 0 0 C   | SRL E<br>2 8<br>Z 0 0 C   | SRL H<br>2 8<br>Z 0 0 C   | SRL (HL)<br>2 16<br>Z 0 0 C   | SRL A<br>2 8<br>Z 0 0 C      |                          |
| 4x | BIT 0,B<br>2 8<br>Z 0 1 - | BIT 0,C<br>2 8<br>Z 0 1 - | BIT 0,D<br>2 8<br>Z 0 1 - | BIT 0,E<br>2 8<br>Z 0 1 - | BIT 0,H<br>2 8<br>Z 0 1 - | BIT 0,L<br>2 8<br>Z 0 1 - | BIT 0,(HL)<br>2 16<br>Z 0 1 - | BIT 0,A<br>2 8<br>Z 0 1 - | BIT 1,B<br>2 8<br>Z 0 1 - | BIT 1,C<br>2 8<br>Z 0 1 - | BIT 1,D<br>2 8<br>Z 0 1 - | BIT 1,E<br>2 8<br>Z 0 1 - | BIT 1,H<br>2 8<br>Z 0 1 - | BIT 1,(HL)<br>2 16<br>Z 0 1 - | BIT 1,A<br>2 8<br>Z 0 1 -    |                          |
| 5x | BIT 2,B<br>2 8<br>Z 0 1 - | BIT 2,C<br>2 8<br>Z 0 1 - | BIT 2,D<br>2 8<br>Z 0 1 - | BIT 2,E<br>2 8<br>Z 0 1 - | BIT 2,H<br>2 8<br>Z 0 1 - | BIT 2,L<br>2 8<br>Z 0 1 - | BIT 2,(HL)<br>2 16<br>Z 0 1 - | BIT 2,A<br>2 8<br>Z 0 1 - | BIT 3,B<br>2 8<br>Z 0 1 - | BIT 3,C<br>2 8<br>Z 0 1 - | BIT 3,D<br>2 8<br>Z 0 1 - | BIT 3,E<br>2 8<br>Z 0 1 - | BIT 3,H<br>2 8<br>Z 0 1 - | BIT 3,(HL)<br>2 16<br>Z 0 1 - | BIT 3,A<br>2 8<br>Z 0 1 -    |                          |
| 6x | BIT 4,B<br>2 8<br>Z 0 1 - | BIT 4,C<br>2 8<br>Z 0 1 - | BIT 4,D<br>2 8<br>Z 0 1 - | BIT 4,E<br>2 8<br>Z 0 1 - | BIT 4,H<br>2 8<br>Z 0 1 - | BIT 4,L<br>2 8<br>Z 0 1 - | BIT 4,(HL)<br>2 16<br>Z 0 1 - | BIT 4,A<br>2 8<br>Z 0 1 - | BIT 5,B<br>2 8<br>Z 0 1 - | BIT 5,C<br>2 8<br>Z 0 1 - | BIT 5,D<br>2 8<br>Z 0 1 - | BIT 5,E<br>2 8<br>Z 0 1 - | BIT 5,H<br>2 8<br>Z 0 1 - | BIT 5,(HL)<br>2 16<br>Z 0 1 - | BIT 5,A<br>2 8<br>Z 0 1 -    |                          |
| 7x | BIT 6,B<br>2 8<br>Z 0 1 - | BIT 6,C<br>2 8<br>Z 0 1 - | BIT 6,D<br>2 8<br>Z 0 1 - | BIT 6,E<br>2 8<br>Z 0 1 - | BIT 6,H<br>2 8<br>Z 0 1 - | BIT 6,L<br>2 8<br>Z 0 1 - | BIT 6,(HL)<br>2 16<br>Z 0 1 - | BIT 6,A<br>2 8<br>Z 0 1 - | BIT 7,B<br>2 8<br>Z 0 1 - | BIT 7,C<br>2 8<br>Z 0 1 - | BIT 7,D<br>2 8<br>Z 0 1 - | BIT 7,E<br>2 8<br>Z 0 1 - | BIT 7,H<br>2 8<br>Z 0 1 - | BIT 7,(HL)<br>2 16<br>Z 0 1 - | BIT 7,A<br>2 8<br>Z 0 1 -    |                          |
| 8x | RES 0,B<br>2 8<br>- - - - | RES 0,C<br>2 8<br>- - - - | RES 0,D<br>2 8<br>- - - - | RES 0,E<br>2 8<br>- - - - | RES 0,H<br>2 8<br>- - - - | RES 0,L<br>2 8<br>- - - - | RES 0,(HL)<br>2 16<br>- - - - | RES 0,A<br>2 8<br>- - - - | RES 1,B<br>2 8<br>- - - - | RES 1,C<br>2 8<br>- - - - | RES 1,D<br>2 8<br>- - - - | RES 1,E<br>2 8<br>- - - - | RES 1,H<br>2 8<br>- - - - | RES 1,(HL)<br>2 16<br>- - - - | RES 1,A<br>2 8<br>- - - -    |                          |
| 9x | RES 2,B<br>2 8<br>- - - - | RES 2,C<br>2 8<br>- - - - | RES 2,D<br>2 8<br>- - - - | RES 2,E<br>2 8<br>- - - - | RES 2,H<br>2 8<br>- - - - | RES 2,L<br>2 8<br>- - - - | RES 2,(HL)<br>2 16<br>- - - - | RES 2,A<br>2 8<br>- - - - | RES 3,B<br>2 8<br>- - - - | RES 3,C<br>2 8<br>- - - - | RES 3,D<br>2 8<br>- - - - | RES 3,E<br>2 8<br>- - - - | RES 3,H<br>2 8<br>- - - - | RES 3,(HL)<br>2 16<br>- - - - | RES 3,A<br>2 8<br>- - - -    |                          |
| Ax | RES 4,B<br>2 8<br>- - - - | RES 4,C<br>2 8<br>- - - - | RES 4,D<br>2 8<br>- - - - | RES 4,E<br>2 8<br>- - - - | RES 4,H<br>2 8<br>- - - - | RES 4,L<br>2 8<br>- - - - | RES 4,(HL)<br>2 16<br>- - - - | RES 4,A<br>2 8<br>- - - - | RES 5,B<br>2 8<br>- - - - | RES 5,C<br>2 8<br>- - - - | RES 5,D<br>2 8<br>- - - - | RES 5,E<br>2 8<br>- - - - | RES 5,H<br>2 8<br>- - - - | RES 5,(HL)<br>2 16<br>- - - - | RES 5,A<br>2 8<br>- - - -    |                          |
| Bx | RES 6,B<br>2 8<br>- - - - | RES 6,C<br>2 8<br>- - - - | RES 6,D<br>2 8<br>- - - - | RES 6,E<br>2 8<br>- - - - | RES 6,H<br>2 8<br>- - - - | RES 6,L<br>2 8<br>- - - - | RES 6,(HL)<br>2 16<br>- - - - | RES 6,A<br>2 8<br>- - - - | RES 7,B<br>2 8<br>- - - - | RES 7,C<br>2 8<br>- - - - | RES 7,D<br>2 8<br>- - - - | RES 7,E<br>2 8<br>- - - - | RES 7,H<br>2 8<br>- - - - | RES 7,(HL)<br>2 16<br>- - - - | RES 7,A<br>2 8<br>- - - -    |                          |
| Cx | SET 0,B<br>2 8<br>- - - - | SET 0,C<br>2 8<br>- - - - | SET 0,D<br>2 8<br>- - - - | SET 0,E<br>2 8<br>- - - - | SET 0,H<br>2 8<br>- - - - | SET 0,L<br>2 8<br>- - - - | SET 0,(HL)<br>2 16<br>- - - - | SET 0,A<br>2 8<br>- - - - | SET 1,B<br>2 8<br>- - - - | SET 1,C<br>2 8<br>- - - - | SET 1,D<br>2 8<br>- - - - | SET 1,E<br>2 8<br>- - - - | SET 1,H<br>2 8<br>- - - - | SET 1,(HL)<br>2 16<br>- - - - | SET 1,A<br>2 8<br>- - - -    |                          |
| Dx | SET 2,B<br>2 8<br>- - - - | SET 2,C<br>2 8<br>- - - - | SET 2,D<br>2 8<br>- - - - | SET 2,E<br>2 8<br>- - - - | SET 2,H<br>2 8<br>- - - - | SET 2,L<br>2 8<br>- - - - | SET 2,(HL)<br>2 16<br>- - - - | SET 2,A<br>2 8<br>- - - - | SET 3,B<br>2 8<br>- - - - | SET 3,C<br>2 8<br>- - - - | SET 3,D<br>2 8<br>- - - - | SET 3,E<br>2 8<br>- - - - | SET 3,H<br>2 8<br>- - - - | SET 3,(HL)<br>2 16<br>- - - - | SET 3,A<br>2 8<br>- - - -    |                          |
| Ex | SET 4,B<br>2 8<br>- - - - | SET 4,C<br>2 8<br>- - - - | SET 4,D<br>2 8<br>- - - - | SET 4,E<br>2 8<br>- - - - | SET 4,H<br>2 8<br>- - - - | SET 4,L<br>2 8<br>- - - - | SET 4,(HL)<br>2 16<br>- - - - | SET 4,A<br>2 8<br>- - - - | SET 5,B<br>2 8<br>- - - - | SET 5,C<br>2 8<br>- - - - | SET 5,D<br>2 8<br>- - - - | SET 5,E<br>2 8<br>- - - - | SET 5,H<br>2 8<br>- - - - | SET 5,(HL)<br>2 16<br>- - - - | SET 5,A<br>2 8<br>- - - -    |                          |
| Fx | SET 6,B<br>2 8<br>- - - - | SET 6,C<br>2 8<br>- - - - | SET 6,D<br>2 8<br>- - - - | SET 6,E<br>2 8<br>- - - - | SET 6,H<br>2 8<br>- - - - | SET 6,L<br>2 8<br>- - - - | SET 6,(HL)<br>2 16<br>- - - - | SET 6,A<br>2 8<br>- - - - | SET 7,B<br>2 8<br>- - - - | SET 7,C<br>2 8<br>- - - - | SET 7,D<br>2 8<br>- - - - | SET 7,E<br>2 8<br>- - - - | SET 7,H<br>2 8<br>- - - - | SET 7,(HL)<br>2 16<br>- - - - | SET 7,A<br>2 8<br>- - - -    |                          |

## Supported Core Features

| Registers                   |                      | Load/Store                                                                                                                    | Artih/Logical                                                    | Rotate               | Misc                                                                             |                                       |
|-----------------------------|----------------------|-------------------------------------------------------------------------------------------------------------------------------|------------------------------------------------------------------|----------------------|----------------------------------------------------------------------------------|---------------------------------------|
| A                           | F                    | ld r8, d8<br>ld r8, r8<br>ld a, (bc)<br>ld a, (de)<br>ld a, (a16)*<br>ld (bc), a<br>ld (de), a<br>ld (a16), a*<br>ld r16, d16 | add<br>adc<br>sub<br>sbc<br>and<br>xor<br>or<br>cp<br>inc<br>dec | a, d8<br>a, r8<br>r8 | rlca<br>rla<br>rrca<br>rra                                                       | ccf<br>scf<br>nop<br>halt<br>di<br>ei |
| H                           | L                    |                                                                                                                               |                                                                  |                      | Control Flow                                                                     |                                       |
| SP                          |                      |                                                                                                                               |                                                                  |                      | jp a16<br>jp hl<br>jp f, a16<br>call a16<br>call f, a16<br>ret<br>ret f<br>rst n |                                       |
| PC                          |                      |                                                                                                                               |                                                                  |                      |                                                                                  |                                       |
| r8                          | r16                  | Stack                                                                                                                         |                                                                  |                      |                                                                                  |                                       |
| a (hl)<br>b c<br>d e<br>h l | bc<br>de<br>hl<br>sp | ld sp, hl<br>push r16<br>pop r16                                                                                              | daa<br>cpl<br>add hl, r16<br>inc r16<br>dec r16                  |                      |                                                                                  |                                       |

\* different opcode than 8080

```
public byte ExecuteNextOperation()

    if (IsHalted)
    {
        return 1;
    }

    var opCode = MemoryController.ReadByte(address: Register.PC);
    var operationBlock = (opCode & 0b11000000) >> 6;

    var operationSize = operationBlock switch
    {
        0x0 => ExecuteBlock0(ref opCode),
        0x1 => ExecuteBlock1(ref opCode),
        0x2 => ExecuteBlock2(ref opCode),
        0x3 => ExecuteBlock3(ref opCode),
        _ => throw new NotImplementedException($"Operation {opCode:X2} not implemented")
    };

    if (Register.IMEPending && opCode != 0xFB)
    {
        Register.IMEPending = false;
        Register.InterruptsMasterEnabled = true;
    }

    Register.PC += operationSize.instructionBytesLength;

    return (byte)(operationSize.durationTStates + (interruptHandled ? 5 : 0));
}
```



# CPU - Przerwania

- “**Zdarzenia**” obsługiwane jak najszybciej to możliwe
- Gameboy posiada zamkniętą listę 5 przerwań
  - 1 związane z Joypadem (**klasyk**)
  - 2 związane z cyklem życia LCD i PPU
  - 1 z timerem
  - 1 z portem szeregowym
- Obsługę zdarzenia definiuje twórca programu umieszczając swoją procedurę pod jednym z adresów: 64, 72, 80, 88, 96

# Liczniki (ang. Timers) - **TIMA, DIV**



# DIV Timer

```
public class DivTimer(MemoryController memoryController)
{
    private int _dividerCycleCounter;

     1 usage
    internal void CheckAndIncrementTimer(ref byte tStates)
    {
        _dividerCycleCounter += tStates;

        if (_dividerCycleCounter >= Cycles.DividerCycles)
        {
            _dividerCycleCounter -= Cycles.DividerCycles;
            memoryController.IncrementByte(Constants.DIVRegister);
        }
    }
}
```

# Mapowanie pamięci



# Mapowanie pamięci

```
public byte[] MemorySpace = new byte[65535]; //64 KB
```

# Tetris



# Pokemon Red

Exception Unhandled



**System.ArgumentOutOfRangeException:** 'Index was out of range. Must be non-negative and less than the size of the collection. (Parameter 'index')

© Tutlane.com

[Show Call Stack](#) | [View Details](#) | [Copy Details](#) | [Start Live Share session...](#)

► [Exception Settings](#)



# Tak wygląda Tetris (32 KB)...



Źródło: <https://b13rg.github.io/Gameboy-Bank-Switching/>



# Ale Pokemony Red (2 MB) już tak...



Źródło: <https://b13rg.github.io/Gameboy-Bank-Switching/>



# MBC - Memory Bank Controller

- **MBC** pozwala na obsługę **wielu banków pamięci** na tym samym adresie
- Dodatkowe funkcje:
  - sterowanie **zegarem czasu rzeczywistego**
  - **zapis save'ów**
  - obsługa urządzeń (kamera, czujnik ruchu, wibracje)

# RTC - Zegar czasu rzeczywistego



# Mapowanie pamięci, ale tym razem z MBC



# Mapowanie pamięci w kodzie

```
private void InitializeMemoryMap()
{
    for (int i = 0; i <= 65535; i++)
    {
        _memoryMap[i] = i switch
        {
            <= BankAddress.RomBankNnEnd => RomBankNn,
            <= BankAddress.VramEnd => Vram,
            <= BankAddress.ExternalRamEnd => RomBankNn,
            <= BankAddress.Wram0End => Wram0,
            <= BankAddress.Wram1End => Wram1,
            <= 0xFFFF => Wram0, //Echo of Wram0
            <= 0xFDFF => Wram1, //Echo of Wram1
            <= BankAddress.OamEnd => Oam,
            <= BankAddress.NotUsableEnd => NotUsable,
            <= BankAddress.IoRegistersEnd => IoRegisters,
            <= BankAddress.HRamEnd => HRam,
            _ => InterruptEnableRegister
        };
    }
}
```



# PPU - Pixel Processing Unit

- **Renderuje** z pamięci VRAM **kafelki (ang. Tiles)**, a nie pojedyncze piksele
  - **1 Tile = 8x8 pikseli**
  - Oszczędność pamięci VRAM!
- Rysuje 3 typy rzeczy na ekranie
  - **Background** (tło)
  - **Window** (okno)
  - **Objects** (obiekty częściej **Sprites**)
- **Rysowanie** odbywa się **rzęd po rzędzie! 144 linii** (ang. **scanlines**) na jedną klatkę

# Tiles / Kafelki zamiast pikseli



Źródło: <https://nesdoug.com/>



# Super Mario na konsoli NES/Pegasus



Źródło: <https://nesdoug.com/>



# Rysowanie tła (Background)



Źródło: <https://hacktix.github.io/GBEDG/ppu/>



Background Tilemap:



Output:



# Rysowanie okna (Window)



Źródło: <https://hacktix.github.io/GBEDG/ppu/>



NIDORANG

L3

HP:



PIKACHU

L50

HP:

135/135



PIKACHU

used THUNDERBOLT!



# Sprites / Objects



Źródło: <https://nesdoug.com/>



# Cykl życia PPU





Źródło: <https://gbdev.io/>





# APU - Audio Processing Unit

- **PPU - Zbiór statycznych stanów**
- **APU - Zbiór timerów (+ stanów)**
- **4 niezależne kanały**
  - Square Wave 1 - Generator Fali Prostokątnej
  - Square Wave 2 - Generator Fali Prostokątnej
  - Wave - Własne próbki
  - Noise - Generator pseudolosowego szumu

# Architecture



# APU - Square 1/2

- Generator fali prostokątnej
- DAC + HPF
- Modulacja:
  - Amplitudy
  - Amplitudy w czasie
  - Częstotliwości
  - Czasu wykonania
  - (*Square 1*) Częstotliwości w czasie tzw. Envelope



# APU - Wave 3

- Generator dowolnie zaprogramowanej fali
- 32 próbki po 4 bity (0-15)
- Modulacja:
  - Amplitudy (głośność)
  - Częstotliwości
  - Czasu wykonania



# APU - Noise 4

- Generator pseudolosowego szumu
- Efekty perkusyjne/dźwiękowe





**Szymon Sandura**

.NET Senior Developer / Tech Lead

[szymon.sandura@iteo.com](mailto:szymon.sandura@iteo.com)



