Compare commits
	
		
			1 Commits
		
	
	
		
			apu-things
			...
			a590048204
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| a590048204 | 
							
								
								
									
										58
									
								
								.github/workflows/main.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										58
									
								
								.github/workflows/main.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,58 +0,0 @@ | ||||
| name: Nightly | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     paths: | ||||
|       - "**.zig" | ||||
|     branches: | ||||
|       - main | ||||
|   schedule: | ||||
|     - cron: '0 0 * * *' | ||||
|   workflow_dispatch: | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest, windows-latest, macos-latest] | ||||
|     runs-on: ${{matrix.os}} | ||||
|     steps: | ||||
|       - uses: goto-bus-stop/setup-zig@v2 | ||||
|         with: | ||||
|           version: master | ||||
|       - name: prepare-linux | ||||
|         if: runner.os == 'Linux' | ||||
|         run: | | ||||
|             sudo apt-get update | ||||
|             sudo apt-get install libsdl2-dev | ||||
|       - name: prepare-windows | ||||
|         if: runner.os == 'Windows' | ||||
|         run: | | ||||
|             vcpkg integrate install | ||||
|             vcpkg install sdl2:x64-windows | ||||
|             git config --global core.autocrlf false | ||||
|       - name: prepare-macos | ||||
|         if: runner.os == 'macOS' | ||||
|         run: | | ||||
|             brew install sdl2 | ||||
|       - uses: actions/checkout@v3 | ||||
|         with: | ||||
|           submodules: true | ||||
|       - name: build  | ||||
|         run: zig build -Drelease-safe | ||||
|       - name: upload | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         with: | ||||
|           name: zba-${{matrix.os}} | ||||
|           path: zig-out/bin | ||||
|   lint: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps:  | ||||
|       - uses: actions/checkout@v3 | ||||
|         with: | ||||
|           submodules: true | ||||
|       - uses: goto-bus-stop/setup-zig@v2 | ||||
|         with: | ||||
|           version: master | ||||
|       - run: zig fmt src/**/*.zig | ||||
|    | ||||
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,7 @@ | ||||
| /.vscode | ||||
| /bin | ||||
| **/zig-cache | ||||
| **/zig-out | ||||
| /zig-cache | ||||
| /zig-out | ||||
| /docs | ||||
| **/*.log | ||||
| **/*.bin | ||||
|   | ||||
							
								
								
									
										6
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							| @@ -7,9 +7,3 @@ | ||||
| [submodule "lib/known-folders"] | ||||
| 	path = lib/known-folders | ||||
| 	url = https://github.com/ziglibs/known-folders | ||||
| [submodule "lib/zig-datetime"] | ||||
| 	path = lib/zig-datetime | ||||
| 	url = https://github.com/frmdstryr/zig-datetime | ||||
| [submodule "lib/zig-toml"] | ||||
| 	path = lib/zig-toml | ||||
| 	url = https://github.com/aeronavery/zig-toml | ||||
|   | ||||
							
								
								
									
										125
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										125
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,116 +1,53 @@ | ||||
| # ZBA (working title) | ||||
| An in-progress Gameboy Advance Emulator written in Zig ⚡! | ||||
|  | ||||
| A Game Boy Advance Emulator written in Zig ⚡! | ||||
|  | ||||
| ## Scope | ||||
|  | ||||
| I'm hardly the first to write a Game Boy Advance Emulator nor will I be the last. This project isn't going to compete with the GOATs like [mGBA](https://github.com/mgba-emu) or [NanoBoyAdvance](https://github.com/nba-emu/NanoBoyAdvance). There aren't any interesting ideas either like in [DSHBA](https://github.com/DenSinH/DSHBA). | ||||
|  | ||||
| This is a simple (read: incomplete) for-fun long-term project. I hope to get "mostly there", which to me means that I'm not missing any major hardware features and the set of possible improvements would be in memory timing or in UI/UX. With respect to that goal, here's what's outstanding: | ||||
|  | ||||
| ### TODO | ||||
|  | ||||
| - [ ] Affine Sprites | ||||
| - [ ] Windowing (see [this branch](https://git.musuka.dev/paoda/zba/src/branch/window)) | ||||
| - [ ] Audio Resampler (Having issues with SDL2's) | ||||
| - [ ] Immediate Mode GUI | ||||
| - [ ] Refactoring for easy-ish perf boosts | ||||
|  | ||||
| ## Usage | ||||
|  | ||||
| As it currently exists, ZBA is run from the terminal. In your console of choice, type `./zba --help` to see what you can do. | ||||
|  | ||||
| I typically find myself typing `./zba -b ./bin/bios.bin ./bin/test/suite.gba` to see how badly my "cool new feature" broke everything else. | ||||
|  | ||||
| Need a BIOS? Why not try using the open-source [Cult-Of-GBA BIOS](https://github.com/Cult-of-GBA/BIOS) written by [fleroviux](https://github.com/fleroviux) and [DenSinH](https://github.com/DenSinH)? | ||||
|  | ||||
| Finally it's worth noting that ZBA uses a TOML config file it'll store in your OS's data directory. See `example.toml` to learn about the defaults and what exactly you can mess around with. | ||||
|  | ||||
| ## Tests | ||||
|  | ||||
| GBA Tests | [jsmolka](https://github.com/jsmolka/) | ||||
| --- | --- | ||||
| `arm.gba`,  `thumb.gba` | PASS | ||||
| `memory.gba`, `bios.gba` | PASS | ||||
| `flash64.gba`, `flash128.gba` | PASS | ||||
| `sram.gba` | PASS | ||||
| `none.gba` | PASS | ||||
| `hello.gba`, `shades.gba`, `stripes.gba` | PASS | ||||
| `nes.gba` | PASS | ||||
|  | ||||
| GBARoms | [DenSinH](https://github.com/DenSinH/) | ||||
| --- | --- | ||||
| `eeprom-test`, `flash-test` | PASS | ||||
| `midikey2freq` | PASS | ||||
| `swi-tests-random` | FAIL | ||||
|  | ||||
| gba_tests | [destoer](https://github.com/destoer/) | ||||
| --- | --- | ||||
| `cond_invalid.gba` | PASS | ||||
| `dma_priority.gba` | PASS | ||||
| `hello_world.gba` | PASS | ||||
| `if_ack.gba` | PASS | ||||
| `line_timing.gba` | FAIL | ||||
| `lyc_midline.gba` | FAIL | ||||
| `window_midframe.gba` | FAIL | ||||
|  | ||||
| GBA Test Collection | [ladystarbreeze](https://github.com/ladystarbreeze) | ||||
| --- | --- | ||||
| `retAddr.gba` | PASS | ||||
| `helloWorld.gba` | PASS | ||||
| `helloAudio.gba` | PASS | ||||
|  | ||||
| FuzzARM | [DenSinH](https://github.com/DenSinH/) | ||||
| --- | --- | ||||
| `main.gba` | PASS | ||||
|  | ||||
| arm7wrestler GBA Fixed | [destoer](https://github.com/destoer) | ||||
| --- | --- | ||||
| `armwrestler-gba-fixed.gba` | PASS | ||||
| ## Tests  | ||||
| - [jsmolka GBA Test Collection](https://github.com/jsmolka/gba-tests) | ||||
|     - [x] `arm.gba` and `thumb.gba` | ||||
|     - [x] `flash64.gba`, `flash128.gba`, `none.gba`, and `sram.gba` | ||||
|     - [x] `hello.gba`, `shades.gba`, and `stripes.gba` | ||||
|     - [x] `memory.gba` | ||||
|     - [ ] `bios.gba` | ||||
|     - [ ] `nes.gba` | ||||
| - [x] [`armwrestler-gba-fixed.gba`](https://github.com/destoer/armwrestler-gba-fixed) | ||||
| - [x] [FuzzARM](https://github.com/DenSinH/FuzzARM) | ||||
|  | ||||
| ## Resources | ||||
|  | ||||
| - [GBATEK](https://problemkaputt.de/gbatek.htm) | ||||
| - [TONC](https://coranac.com/tonc/text/toc.htm) | ||||
| - [ARM Architecture Reference Manual](https://www.intel.com/content/dam/www/programmable/us/en/pdfs/literature/third-party/ddi0100e_arm_arm.pdf) | ||||
| - [ARM7TDMI Data Sheet](https://www.dca.fee.unicamp.br/cursos/EA871/references/ARM/ARM7TDMIDataSheet.pdf) | ||||
| * [GBATEK](https://problemkaputt.de/gbatek.htm) | ||||
| * [TONC](https://coranac.com/tonc/text/toc.htm) | ||||
| * [ARM Architecture Reference Manual](https://www.intel.com/content/dam/www/programmable/us/en/pdfs/literature/third-party/ddi0100e_arm_arm.pdf) | ||||
| * [ARM7TDMI Data Sheet](https://www.dca.fee.unicamp.br/cursos/EA871/references/ARM/ARM7TDMIDataSheet.pdf) | ||||
|  | ||||
| ## Compiling | ||||
|  | ||||
| Most recently built on Zig [0.11.0-dev.368+1829b6eab](https://github.com/ziglang/zig/tree/1829b6eab) | ||||
| Most recently built on Zig [0.10.0-dev.1741+d2681d253](https://github.com/ziglang/zig/tree/d2681d253) | ||||
|  | ||||
| ### Dependencies | ||||
| * [SDL.zig](https://github.com/MasterQ32/SDL.zig) | ||||
|     * [SDL2](https://www.libsdl.org/download-2.0.php) | ||||
| * [zig-clap](https://github.com/Hejsil/zig-clap) | ||||
| * [known-folders](https://github.com/ziglibs/known-folders) | ||||
| * [`bitfields.zig`](https://github.com/FlorenceOS/Florence/blob/f6044db788d35d43d66c1d7e58ef1e3c79f10d6f/lib/util/bitfields.zig) | ||||
|  | ||||
| Dependency | Source | ||||
| --- | --- | ||||
| SDL.zig | <https://github.com/MasterQ32/SDL.zig> | ||||
| zig-clap | <https://github.com/Hejsil/zig-clap> | ||||
| known-folders | <https://github.com/ziglibs/known-folders> | ||||
| zig-toml | <https://github.com/aeronavery/zig-toml> | ||||
| zig-datetime | <https://github.com/frmdstryr/zig-datetime> | ||||
| `bitfields.zig` | [https://github.com/FlorenceOS/Florence](https://github.com/FlorenceOS/Florence/blob/aaa5a9e568/lib/util/bitfields.zig) | ||||
| `gl.zig` | <https://github.com/MasterQ32/zig-opengl> | ||||
| `bitfields.zig` from [FlorenceOS](https://github.com/FlorenceOS) is included under `lib/util/bitfield.zig`. | ||||
|  | ||||
| Use `git submodule update --init` from the project root to pull the git submodules `SDL.zig`, `zig-clap`, `known-folders`, `zig-toml` and `zig-datetime` | ||||
| Use `git submodule update --init` from the project root to pull the git submodules `SDL.zig`, `zig-clap`, and `known-folders` | ||||
|  | ||||
| Be sure to provide SDL2 using: | ||||
| Be sure to provide SDL2 using:  | ||||
| * Linux: Your distro's package manager | ||||
| * MacOS: ¯\\\_(ツ)_/¯ | ||||
| * Windows: [`vcpkg`](https://github.com/Microsoft/vcpkg) (install `sdl2:x64-windows`) | ||||
|  | ||||
| - Linux: Your distro's package manager | ||||
| - MacOS: ¯\\\_(ツ)_/¯ | ||||
| - Windows: [`vcpkg`](https://github.com/Microsoft/vcpkg) (install `sdl2:x64-windows`) | ||||
| `SDL.zig` will provide a helpful compile error if the zig compiler is unable to find SDL2.  | ||||
|  | ||||
| `SDL.zig` will provide a helpful compile error if the zig compiler is unable to find SDL2. | ||||
|  | ||||
| Once you've got all the dependencies, execute `zig build -Drelease-fast`. The executable is located at `zig-out/bin/`. | ||||
| Once you've got all the dependencies, execute `zig build -Drelease-fast`. The executable is located at `zig-out/bin/`.  | ||||
|  | ||||
| ## Controls | ||||
|  | ||||
| Key | Button | ||||
| --- | --- | ||||
| <kbd>X</kbd> | A | ||||
| <kbd>Z</kbd> | B | ||||
| <kbd>A</kbd> | L | ||||
| <kbd>S</kbd> | R | ||||
| <kbd>A</kbd> | Left Shoulder | ||||
| <kbd>S</kbd> | Right Shoulder | ||||
| <kbd>Return</kbd> | Start | ||||
| <kbd>RShift</kbd> | Select | ||||
| Arrow Keys | D-Pad | ||||
|   | ||||
							
								
								
									
										23
									
								
								build.zig
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								build.zig
									
									
									
									
									
								
							| @@ -1,15 +1,7 @@ | ||||
| const std = @import("std"); | ||||
| const builtin = @import("builtin"); | ||||
| const Sdk = @import("lib/SDL.zig/Sdk.zig"); | ||||
|  | ||||
| pub fn build(b: *std.build.Builder) void { | ||||
|     // Minimum Zig Version | ||||
|     const min_ver = std.SemanticVersion.parse("0.11.0-dev.323+30eb2a175") catch return; // https://github.com/ziglang/zig/commit/30eb2a175 | ||||
|     if (builtin.zig_version.order(min_ver).compare(.lt)) { | ||||
|         std.log.err("{s}", .{b.fmt("Zig v{} does not meet the minimum version requirement. (Zig v{})", .{ builtin.zig_version, min_ver })}); | ||||
|         std.os.exit(1); | ||||
|     } | ||||
|  | ||||
|     // Standard target options allows the person running `zig build` to choose | ||||
|     // what target to build for. Here we do not override the defaults, which | ||||
|     // means any target is allowed, and the default is native. Other options | ||||
| @@ -21,15 +13,10 @@ pub fn build(b: *std.build.Builder) void { | ||||
|     const mode = b.standardReleaseOptions(); | ||||
|  | ||||
|     const exe = b.addExecutable("zba", "src/main.zig"); | ||||
|     exe.setMainPkgPath("."); // Necessary so that src/main.zig can embed example.toml | ||||
|     exe.setTarget(target); | ||||
|  | ||||
|      | ||||
|     // Known Folders (%APPDATA%, XDG, etc.) | ||||
|     exe.addPackagePath("known_folders", "lib/known-folders/known-folders.zig"); | ||||
|  | ||||
|     // DateTime Library | ||||
|     exe.addPackagePath("datetime", "lib/zig-datetime/src/main.zig"); | ||||
|  | ||||
|     // Bitfield type from FlorenceOS: https://github.com/FlorenceOS/ | ||||
|     // exe.addPackage(.{ .name = "bitfield", .path = .{ .path = "lib/util/bitfield.zig" } }); | ||||
|     exe.addPackagePath("bitfield", "lib/util/bitfield.zig"); | ||||
| @@ -37,17 +24,13 @@ pub fn build(b: *std.build.Builder) void { | ||||
|     // Argument Parsing Library | ||||
|     exe.addPackagePath("clap", "lib/zig-clap/clap.zig"); | ||||
|  | ||||
|     // TOML Library | ||||
|     exe.addPackagePath("toml", "lib/zig-toml/src/toml.zig"); | ||||
|  | ||||
|     // OpenGL 3.3 Bindings | ||||
|     exe.addPackagePath("gl", "lib/gl.zig"); | ||||
|  | ||||
|     // Zig SDL Bindings: https://github.com/MasterQ32/SDL.zig | ||||
|     const sdk = Sdk.init(b); | ||||
|     sdk.link(exe, .dynamic); | ||||
|  | ||||
|     exe.addPackage(sdk.getNativePackage("sdl2")); | ||||
|  | ||||
|     exe.setTarget(target); | ||||
|     exe.setBuildMode(mode); | ||||
|     exe.install(); | ||||
|  | ||||
|   | ||||
							
								
								
									
										25
									
								
								example.toml
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								example.toml
									
									
									
									
									
								
							| @@ -1,25 +0,0 @@ | ||||
| [Host] | ||||
| # Using nearest-neighbour scaling, how many times the native resolution  | ||||
| # of the game bow should the screen be? | ||||
| win_scale = 3 | ||||
| # Enable VSYNC on the UI thread | ||||
| vsync = true | ||||
| # Mute ZBA | ||||
| mute = false | ||||
|  | ||||
| [Guest] | ||||
| # Sync Emulation to Audio  | ||||
| audio_sync = true | ||||
| # Sync Emulation to Video | ||||
| video_sync = true | ||||
| # Force RTC support | ||||
| force_rtc = false | ||||
| # Skip BIOS | ||||
| skip_bios = false | ||||
|  | ||||
| [Debug] | ||||
| # Enable detailed CPU logs | ||||
| cpu_trace = false | ||||
| # When false and builtin.mode == .Debug, ZBA will panic | ||||
| # on unknown I/O reads | ||||
| unhandled_io = true | ||||
 Submodule lib/SDL.zig updated: 00b4356885...47b8ab43f3
									
								
							
							
								
								
									
										5028
									
								
								lib/gl.zig
									
									
									
									
									
								
							
							
						
						
									
										5028
									
								
								lib/gl.zig
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							 Submodule lib/known-folders updated: 24845b0103...9db1b99219
									
								
							| @@ -14,7 +14,7 @@ fn PtrCastPreserveCV(comptime T: type, comptime PtrToT: type, comptime NewT: typ | ||||
| fn BitType(comptime FieldType: type, comptime ValueType: type, comptime shamt: usize) type { | ||||
|     const self_bit: FieldType = (1 << shamt); | ||||
|  | ||||
|     return extern struct { | ||||
|     return struct { | ||||
|         bits: Bitfield(FieldType, shamt, 1), | ||||
|  | ||||
|         pub fn set(self: anytype) void { | ||||
| @@ -63,7 +63,7 @@ pub fn Bitfield(comptime FieldType: type, comptime shamt: usize, comptime num_bi | ||||
|  | ||||
|     const ValueType = std.meta.Int(.unsigned, num_bits); | ||||
|  | ||||
|     return extern struct { | ||||
|     return struct { | ||||
|         dummy: FieldType, | ||||
|  | ||||
|         fn field(self: anytype) PtrCastPreserveCV(@This(), @TypeOf(self), FieldType) { | ||||
|   | ||||
 Submodule lib/zig-clap updated: a1b01ffeab...ac5f46541c
									
								
							 Submodule lib/zig-datetime deleted from 932d284521
									
								
							 Submodule lib/zig-toml deleted from 016b8bcf98
									
								
							
							
								
								
									
										186
									
								
								src/Bus.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										186
									
								
								src/Bus.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,186 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const AudioDeviceId = @import("sdl2").SDL_AudioDeviceID; | ||||
| const Arm7tdmi = @import("cpu.zig").Arm7tdmi; | ||||
| const Bios = @import("bus/Bios.zig"); | ||||
| const Ewram = @import("bus/Ewram.zig"); | ||||
| const GamePak = @import("bus/GamePak.zig"); | ||||
| const Io = @import("bus/io.zig").Io; | ||||
| const Iwram = @import("bus/Iwram.zig"); | ||||
| const Ppu = @import("ppu.zig").Ppu; | ||||
| const Apu = @import("apu.zig").Apu; | ||||
| const DmaControllers = @import("bus/dma.zig").DmaControllers; | ||||
| const Timers = @import("bus/timer.zig").Timers; | ||||
| const Scheduler = @import("scheduler.zig").Scheduler; | ||||
| const FilePaths = @import("util.zig").FilePaths; | ||||
|  | ||||
| const io = @import("bus/io.zig"); | ||||
| const Allocator = std.mem.Allocator; | ||||
| const log = std.log.scoped(.Bus); | ||||
|  | ||||
| const rotr = @import("util.zig").rotr; | ||||
| const Self = @This(); | ||||
|  | ||||
| const panic_on_und_bus: bool = false; | ||||
|  | ||||
| pak: GamePak, | ||||
| bios: Bios, | ||||
| ppu: Ppu, | ||||
| apu: Apu, | ||||
| dma: DmaControllers, | ||||
| tim: Timers, | ||||
| iwram: Iwram, | ||||
| ewram: Ewram, | ||||
| io: Io, | ||||
|  | ||||
| cpu: ?*Arm7tdmi, | ||||
| sched: *Scheduler, | ||||
|  | ||||
| pub fn init(alloc: Allocator, sched: *Scheduler, paths: FilePaths) !Self { | ||||
|     return Self{ | ||||
|         .pak = try GamePak.init(alloc, paths.rom, paths.save), | ||||
|         .bios = try Bios.init(alloc, paths.bios), | ||||
|         .ppu = try Ppu.init(alloc, sched), | ||||
|         .apu = Apu.init(), | ||||
|         .iwram = try Iwram.init(alloc), | ||||
|         .ewram = try Ewram.init(alloc), | ||||
|         .dma = DmaControllers.init(), | ||||
|         .tim = Timers.init(sched), | ||||
|         .io = Io.init(), | ||||
|         .cpu = null, | ||||
|         .sched = sched, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: Self) void { | ||||
|     self.iwram.deinit(); | ||||
|     self.ewram.deinit(); | ||||
|     self.pak.deinit(); | ||||
|     self.bios.deinit(); | ||||
|     self.ppu.deinit(); | ||||
| } | ||||
|  | ||||
| pub fn handleDMATransfers(self: *Self) void { | ||||
|     while (self.isDmaRunning()) { | ||||
|         if (self.dma._1.step(self)) continue; | ||||
|         if (self.dma._0.step(self)) continue; | ||||
|         if (self.dma._2.step(self)) continue; | ||||
|         if (self.dma._3.step(self)) continue; | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn isDmaRunning(self: *const Self) bool { | ||||
|     return self.dma._0.active or | ||||
|         self.dma._1.active or | ||||
|         self.dma._2.active or | ||||
|         self.dma._3.active; | ||||
| } | ||||
|  | ||||
| pub fn debugRead(self: *const Self, comptime T: type, address: u32) T { | ||||
|     const cached = self.sched.tick; | ||||
|     defer self.sched.tick = cached; | ||||
|  | ||||
|     return self.read(T, address); | ||||
| } | ||||
|  | ||||
| fn readOpenBus(self: *const Self, comptime T: type, address: u32) T { | ||||
|     if (self.cpu.?.cpsr.t.read()) { | ||||
|         log.err("TODO: {} open bus read in THUMB", .{T}); | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     const word = self.debugRead(u32, self.cpu.?.r[15] + 4); | ||||
|     return @truncate(T, rotr(u32, word, 8 * (address & 3))); | ||||
| } | ||||
|  | ||||
| fn readBios(self: *const Self, comptime T: type, address: u32) T { | ||||
|     if (address < Bios.size) return self.bios.read(T, alignAddress(T, address)); | ||||
|  | ||||
|     return self.readOpenBus(T, address); | ||||
| } | ||||
|  | ||||
| pub fn read(self: *const Self, comptime T: type, address: u32) T { | ||||
|     const page = @truncate(u8, address >> 24); | ||||
|     const align_addr = alignAddress(T, address); | ||||
|     self.sched.tick += 1; | ||||
|  | ||||
|     return switch (page) { | ||||
|         // General Internal Memory | ||||
|         0x00 => self.readBios(T, address), | ||||
|         0x02 => self.ewram.read(T, align_addr), | ||||
|         0x03 => self.iwram.read(T, align_addr), | ||||
|         0x04 => io.read(self, T, align_addr), | ||||
|  | ||||
|         // Internal Display Memory | ||||
|         0x05 => self.ppu.palette.read(T, align_addr), | ||||
|         0x06 => self.ppu.vram.read(T, align_addr), | ||||
|         0x07 => self.ppu.oam.read(T, align_addr), | ||||
|  | ||||
|         // External Memory (Game Pak) | ||||
|         0x08...0x0D => self.pak.read(T, align_addr), | ||||
|         0x0E...0x0F => blk: { | ||||
|             const value = self.pak.backup.read(address); | ||||
|  | ||||
|             const multiplier = switch (T) { | ||||
|                 u32 => 0x01010101, | ||||
|                 u16 => 0x0101, | ||||
|                 u8 => 1, | ||||
|                 else => @compileError("Backup: Unsupported read width"), | ||||
|             }; | ||||
|  | ||||
|             break :blk @as(T, value) * multiplier; | ||||
|         }, | ||||
|         else => readOpenBus(self, T, address), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(self: *Self, comptime T: type, address: u32, value: T) void { | ||||
|     const page = @truncate(u8, address >> 24); | ||||
|     const align_addr = alignAddress(T, address); | ||||
|     self.sched.tick += 1; | ||||
|  | ||||
|     switch (page) { | ||||
|         // General Internal Memory | ||||
|         0x00 => self.bios.write(T, align_addr, value), | ||||
|         0x02 => self.ewram.write(T, align_addr, value), | ||||
|         0x03 => self.iwram.write(T, align_addr, value), | ||||
|         0x04 => io.write(self, T, align_addr, value), | ||||
|  | ||||
|         // Internal Display Memory | ||||
|         0x05 => self.ppu.palette.write(T, align_addr, value), | ||||
|         0x06 => self.ppu.vram.write(T, self.ppu.dispcnt, align_addr, value), | ||||
|         0x07 => self.ppu.oam.write(T, align_addr, value), | ||||
|  | ||||
|         // External Memory (Game Pak) | ||||
|         0x08...0x0D => {}, | ||||
|         0x0E...0x0F => { | ||||
|             const rotate_by = switch (T) { | ||||
|                 u32 => address & 3, | ||||
|                 u16 => address & 1, | ||||
|                 u8 => 0, | ||||
|                 else => @compileError("Backup: Unsupported write width"), | ||||
|             }; | ||||
|  | ||||
|             self.pak.backup.write(address, @truncate(u8, rotr(T, value, 8 * rotate_by))); | ||||
|         }, | ||||
|         else => undWrite("Tried to write {} 0x{X:} to 0x{X:0>8}", .{ T, value, address }), | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn alignAddress(comptime T: type, address: u32) u32 { | ||||
|     return switch (T) { | ||||
|         u32 => address & 0xFFFF_FFFC, | ||||
|         u16 => address & 0xFFFF_FFFE, | ||||
|         u8 => address, | ||||
|         else => @compileError("Bus: Invalid read/write type"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| fn undRead(comptime format: []const u8, args: anytype) u8 { | ||||
|     if (panic_on_und_bus) std.debug.panic(format, args) else log.warn(format, args); | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| fn undWrite(comptime format: []const u8, args: anytype) void { | ||||
|     if (panic_on_und_bus) std.debug.panic(format, args) else log.warn(format, args); | ||||
| } | ||||
							
								
								
									
										266
									
								
								src/apu.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										266
									
								
								src/apu.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,266 @@ | ||||
| const std = @import("std"); | ||||
| const SDL = @import("sdl2"); | ||||
| const io = @import("bus/io.zig"); | ||||
| const Arm7tdmi = @import("cpu.zig").Arm7tdmi; | ||||
|  | ||||
| const SoundFifo = std.fifo.LinearFifo(u8, .{ .Static = 0x20 }); | ||||
| const AudioDeviceId = SDL.SDL_AudioDeviceID; | ||||
|  | ||||
| const intToBytes = @import("util.zig").intToBytes; | ||||
| const log = std.log.scoped(.APU); | ||||
|  | ||||
| pub const Apu = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     ch1: ToneSweep, | ||||
|     ch2: Tone, | ||||
|     ch3: Wave, | ||||
|     ch4: Noise, | ||||
|     chA: DmaSound(.A), | ||||
|     chB: DmaSound(.B), | ||||
|  | ||||
|     bias: io.SoundBias, | ||||
|     ch_vol_cnt: io.ChannelVolumeControl, | ||||
|     dma_cnt: io.DmaSoundControl, | ||||
|     cnt: io.SoundControl, | ||||
|  | ||||
|     dev: ?AudioDeviceId, | ||||
|  | ||||
|     pub fn init() Self { | ||||
|         return .{ | ||||
|             .ch1 = ToneSweep.init(), | ||||
|             .ch2 = Tone.init(), | ||||
|             .ch3 = Wave.init(), | ||||
|             .ch4 = Noise.init(), | ||||
|             .chA = DmaSound(.A).init(), | ||||
|             .chB = DmaSound(.B).init(), | ||||
|  | ||||
|             .ch_vol_cnt = .{ .raw = 0 }, | ||||
|             .dma_cnt = .{ .raw = 0 }, | ||||
|             .cnt = .{ .raw = 0 }, | ||||
|             .bias = .{ .raw = 0x0200 }, | ||||
|  | ||||
|             .dev = null, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn attachAudioDevice(self: *Self, dev: AudioDeviceId) void { | ||||
|         self.dev = dev; | ||||
|     } | ||||
|  | ||||
|     pub fn setDmaCnt(self: *Self, value: u16) void { | ||||
|         const new: io.DmaSoundControl = .{ .raw = value }; | ||||
|  | ||||
|         // Reinitializing instead of resetting is fine because | ||||
|         // the FIFOs I'm using are stack allocated and 0x20 bytes big | ||||
|         if (new.sa_reset.read()) self.chA.fifo = SoundFifo.init(); | ||||
|         if (new.sb_reset.read()) self.chB.fifo = SoundFifo.init(); | ||||
|  | ||||
|         self.dma_cnt = new; | ||||
|     } | ||||
|  | ||||
|     pub fn setSoundCntX(self: *Self, value: bool) void { | ||||
|         self.cnt.apu_enable.write(value); | ||||
|     } | ||||
|  | ||||
|     pub fn setSoundCntLLow(self: *Self, byte: u8) void { | ||||
|         self.ch_vol_cnt.raw = (self.ch_vol_cnt.raw & 0xFF00) | byte; | ||||
|     } | ||||
|  | ||||
|     pub fn setSoundCntLHigh(self: *Self, byte: u8) void { | ||||
|         self.ch_vol_cnt.raw = @as(u16, byte) << 8 | (self.ch_vol_cnt.raw & 0xFF); | ||||
|     } | ||||
|  | ||||
|     pub fn setBiasHigh(self: *Self, byte: u8) void { | ||||
|         self.bias.raw = (@as(u16, byte) << 8) | (self.bias.raw & 0xFF); | ||||
|     } | ||||
|  | ||||
|     pub fn handleTimerOverflow(self: *Self, kind: DmaSoundKind, cpu: *Arm7tdmi) void { | ||||
|         if (!self.cnt.apu_enable.read()) return; | ||||
|  | ||||
|         const samples = switch (kind) { | ||||
|             .A => blk: { | ||||
|                 break :blk self.chA.handleTimerOverflow(cpu, self.dma_cnt); | ||||
|             }, | ||||
|             .B => blk: { | ||||
|                 break :blk self.chB.handleTimerOverflow(cpu, self.dma_cnt); | ||||
|             }, | ||||
|         }; | ||||
|  | ||||
|         if (self.dev) |dev| _ = SDL.SDL_QueueAudio(dev, &samples, 2); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const ToneSweep = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     /// NR10 | ||||
|     sweep: io.Sweep, | ||||
|     /// NR11 | ||||
|     duty: io.Duty, | ||||
|     /// NR12 | ||||
|     envelope: io.Envelope, | ||||
|     /// NR13, NR14 | ||||
|     freq: io.Frequency, | ||||
|  | ||||
|     fn init() Self { | ||||
|         return .{ | ||||
|             .sweep = .{ .raw = 0 }, | ||||
|             .duty = .{ .raw = 0 }, | ||||
|             .envelope = .{ .raw = 0 }, | ||||
|             .freq = .{ .raw = 0 }, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn setFreqLow(self: *Self, byte: u8) void { | ||||
|         self.freq.raw = (self.freq.raw & 0xFF00) | byte; | ||||
|     } | ||||
|  | ||||
|     pub fn setFreqHigh(self: *Self, byte: u8) void { | ||||
|         self.freq.raw = (@as(u16, byte) << 8) | (self.freq.raw & 0xFF); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Tone = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     /// NR21 | ||||
|     duty: io.Duty, | ||||
|     /// NR22 | ||||
|     envelope: io.Envelope, | ||||
|     /// NR23, NR24 | ||||
|     freq: io.Frequency, | ||||
|  | ||||
|     fn init() Self { | ||||
|         return .{ | ||||
|             .duty = .{ .raw = 0 }, | ||||
|             .envelope = .{ .raw = 0 }, | ||||
|             .freq = .{ .raw = 0 }, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn setFreqLow(self: *Self, byte: u8) void { | ||||
|         self.freq.raw = (self.freq.raw & 0xFF00) | byte; | ||||
|     } | ||||
|  | ||||
|     pub fn setFreqHigh(self: *Self, byte: u8) void { | ||||
|         self.freq.raw = @as(u16, byte) << 8 | (self.freq.raw & 0xFF); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Wave = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     /// Write-only | ||||
|     /// NR30 | ||||
|     select: io.WaveSelect, | ||||
|     /// NR31 | ||||
|     length: u8, | ||||
|     /// NR32 | ||||
|     vol: io.WaveVolume, | ||||
|     /// NR33, NR34 | ||||
|     freq: io.Frequency, | ||||
|  | ||||
|     fn init() Self { | ||||
|         return .{ | ||||
|             .select = .{ .raw = 0 }, | ||||
|             .vol = .{ .raw = 0 }, | ||||
|             .freq = .{ .raw = 0 }, | ||||
|             .length = 0, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn setFreqLow(self: *Self, byte: u8) void { | ||||
|         self.freq.raw = (self.freq.raw & 0xFF00) | byte; | ||||
|     } | ||||
|  | ||||
|     pub fn setFreqHigh(self: *Self, byte: u8) void { | ||||
|         self.freq.raw = @as(u16, byte) << 8 | (self.freq.raw & 0xFF); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Noise = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     /// Write-only | ||||
|     /// NR41 | ||||
|     len: u6, | ||||
|     /// NR42 | ||||
|     envelope: io.Envelope, | ||||
|     /// NR43 | ||||
|     poly: io.PolyCounter, | ||||
|     /// NR44 | ||||
|     cnt: io.NoiseControl, | ||||
|  | ||||
|     fn init() Self { | ||||
|         return .{ | ||||
|             .len = 0, | ||||
|             .envelope = .{ .raw = 0 }, | ||||
|             .poly = .{ .raw = 0 }, | ||||
|             .cnt = .{ .raw = 0 }, | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| pub fn DmaSound(comptime kind: DmaSoundKind) type { | ||||
|     return struct { | ||||
|         const Self = @This(); | ||||
|  | ||||
|         fifo: SoundFifo, | ||||
|  | ||||
|         kind: DmaSoundKind, | ||||
|  | ||||
|         fn init() Self { | ||||
|             return .{ .fifo = SoundFifo.init(), .kind = kind }; | ||||
|         } | ||||
|  | ||||
|         pub fn push(self: *Self, value: u32) void { | ||||
|             self.fifo.write(&intToBytes(u32, value)) catch {}; | ||||
|         } | ||||
|  | ||||
|         pub fn pop(self: *Self) u8 { | ||||
|             return self.fifo.readItem() orelse 0; | ||||
|         } | ||||
|  | ||||
|         pub fn len(self: *const Self) usize { | ||||
|             return self.fifo.readableLength(); | ||||
|         } | ||||
|  | ||||
|         pub fn handleTimerOverflow(self: *Self, cpu: *Arm7tdmi, cnt: io.DmaSoundControl) [2]u8 { | ||||
|             const sample = self.pop(); | ||||
|  | ||||
|             var left: u8 = 0; | ||||
|             var right: u8 = 0; | ||||
|             var fifo_addr: u32 = undefined; | ||||
|  | ||||
|             switch (kind) { | ||||
|                 .A => { | ||||
|                     const vol = @boolToInt(!cnt.sa_vol.read()); // if unset, vol is 50% | ||||
|                     if (cnt.sa_left_enable.read()) left = sample >> vol; | ||||
|                     if (cnt.sa_right_enable.read()) right = sample >> vol; | ||||
|  | ||||
|                     fifo_addr = 0x0400_00A0; | ||||
|                 }, | ||||
|                 .B => { | ||||
|                     const vol = @boolToInt(!cnt.sb_vol.read()); // if unset, vol is 50% | ||||
|                     if (cnt.sb_left_enable.read()) left = sample >> vol; | ||||
|                     if (cnt.sb_right_enable.read()) right = sample >> vol; | ||||
|  | ||||
|                     fifo_addr = 0x0400_00A4; | ||||
|                 }, | ||||
|             } | ||||
|  | ||||
|             if (self.len() <= 15) { | ||||
|                 cpu.bus.dma._1.enableSoundDma(fifo_addr); | ||||
|                 cpu.bus.dma._2.enableSoundDma(fifo_addr); | ||||
|             } | ||||
|  | ||||
|             return .{ left, right }; | ||||
|         } | ||||
|     }; | ||||
| } | ||||
|  | ||||
| const DmaSoundKind = enum { | ||||
|     A, | ||||
|     B, | ||||
| }; | ||||
							
								
								
									
										52
									
								
								src/bus/Bios.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/bus/Bios.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
| const log = std.log.scoped(.Bios); | ||||
|  | ||||
| /// Size of the BIOS in bytes | ||||
| pub const size = 0x4000; | ||||
| const Self = @This(); | ||||
|  | ||||
| buf: ?[]u8, | ||||
| alloc: Allocator, | ||||
|  | ||||
| pub fn init(alloc: Allocator, maybe_path: ?[]const u8) !Self { | ||||
|     var buf: ?[]u8 = null; | ||||
|     if (maybe_path) |path| { | ||||
|         const file = try std.fs.cwd().openFile(path, .{}); | ||||
|         defer file.close(); | ||||
|         buf = try file.readToEndAlloc(alloc, try file.getEndPos()); | ||||
|     } | ||||
|  | ||||
|     return Self{ | ||||
|         .buf = buf, | ||||
|         .alloc = alloc, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: Self) void { | ||||
|     if (self.buf) |buf| self.alloc.free(buf); | ||||
| } | ||||
|  | ||||
| pub fn read(self: *const Self, comptime T: type, addr: usize) T { | ||||
|     if (self.buf) |buf| { | ||||
|         if (addr > buf.len) { | ||||
|             log.err("Tried to read {} from {X:0>8} (open bus)", .{ T, addr }); | ||||
|             return 0; | ||||
|         } | ||||
|  | ||||
|         return switch (T) { | ||||
|             u32 => (@as(u32, buf[addr + 3]) << 24) | (@as(u32, buf[addr + 2]) << 16) | (@as(u32, buf[addr + 1]) << 8) | (@as(u32, buf[addr])), | ||||
|             u16 => (@as(u16, buf[addr + 1]) << 8) | @as(u16, buf[addr]), | ||||
|             u8 => buf[addr], | ||||
|             else => @compileError("BIOS: Unsupported read width"), | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     std.debug.panic("[BIOS] ZBA tried to read {} from 0x{X:0>8} but not BIOS was present", .{ T, addr }); | ||||
| } | ||||
|  | ||||
| pub fn write(_: *Self, comptime T: type, addr: usize, value: T) void { | ||||
|     @setCold(true); | ||||
|     log.err("Tried to write {} 0x{X:} to 0x{X:0>8} ", .{ T, value, addr }); | ||||
| } | ||||
							
								
								
									
										52
									
								
								src/bus/Ewram.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/bus/Ewram.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
| const ewram_size = 0x40000; | ||||
| const Self = @This(); | ||||
|  | ||||
| buf: []u8, | ||||
| alloc: Allocator, | ||||
|  | ||||
| pub fn init(alloc: Allocator) !Self { | ||||
|     const buf = try alloc.alloc(u8, ewram_size); | ||||
|     std.mem.set(u8, buf, 0); | ||||
|  | ||||
|     return Self{ | ||||
|         .buf = buf, | ||||
|         .alloc = alloc, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: Self) void { | ||||
|     self.alloc.free(self.buf); | ||||
| } | ||||
|  | ||||
| pub fn read(self: *const Self, comptime T: type, address: usize) T { | ||||
|     const addr = address & 0x3FFFF; | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => (@as(u32, self.buf[addr + 3]) << 24) | (@as(u32, self.buf[addr + 2]) << 16) | (@as(u32, self.buf[addr + 1]) << 8) | (@as(u32, self.buf[addr])), | ||||
|         u16 => (@as(u16, self.buf[addr + 1]) << 8) | @as(u16, self.buf[addr]), | ||||
|         u8 => self.buf[addr], | ||||
|         else => @compileError("EWRAM: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(self: *const Self, comptime T: type, address: usize, value: T) void { | ||||
|     const addr = address & 0x3FFFF; | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => { | ||||
|             self.buf[addr + 3] = @truncate(u8, value >> 24); | ||||
|             self.buf[addr + 2] = @truncate(u8, value >> 16); | ||||
|             self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|             self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|         }, | ||||
|         u16 => { | ||||
|             self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|             self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|         }, | ||||
|         u8 => self.buf[addr] = value, | ||||
|         else => @compileError("EWRAM: Unsupported write width"), | ||||
|     }; | ||||
| } | ||||
							
								
								
									
										104
									
								
								src/bus/GamePak.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								src/bus/GamePak.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,104 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Backup = @import("backup.zig").Backup; | ||||
| const Allocator = std.mem.Allocator; | ||||
| const log = std.log.scoped(.GamePak); | ||||
|  | ||||
| const intToBytes = @import("../util.zig").intToBytes; | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| title: [12]u8, | ||||
| buf: []u8, | ||||
| alloc: Allocator, | ||||
| backup: Backup, | ||||
|  | ||||
| pub fn init(alloc: Allocator, rom_path: []const u8, save_path: ?[]const u8) !Self { | ||||
|     const file = try std.fs.cwd().openFile(rom_path, .{}); | ||||
|     defer file.close(); | ||||
|  | ||||
|     const file_buf = try file.readToEndAlloc(alloc, try file.getEndPos()); | ||||
|     const title = parseTitle(file_buf); | ||||
|     const kind = Backup.guessKind(file_buf) orelse .None; | ||||
|  | ||||
|     const pak = Self{ | ||||
|         .buf = file_buf, | ||||
|         .alloc = alloc, | ||||
|         .title = title, | ||||
|         .backup = try Backup.init(alloc, kind, title, save_path), | ||||
|     }; | ||||
|     pak.parseHeader(); | ||||
|  | ||||
|     return pak; | ||||
| } | ||||
|  | ||||
| fn parseHeader(self: *const Self) void { | ||||
|     const title = parseTitle(self.buf); | ||||
|     const code = self.buf[0xAC..0xB0]; | ||||
|     const maker = self.buf[0xB0..0xB2]; | ||||
|     const version = self.buf[0xBC]; | ||||
|  | ||||
|     log.info("Title: {s}", .{title}); | ||||
|     if (version != 0) log.info("Version: {}", .{version}); | ||||
|     log.info("Game Code: {s}", .{code}); | ||||
|     if (lookupMaker(maker)) |c| log.info("Maker: {s}", .{c}) else log.info("Maker Code: {s}", .{maker}); | ||||
| } | ||||
|  | ||||
| fn parseTitle(buf: []u8) [12]u8 { | ||||
|     return buf[0xA0..0xAC].*; | ||||
| } | ||||
|  | ||||
| fn lookupMaker(slice: *const [2]u8) ?[]const u8 { | ||||
|     const id = @as(u16, slice[1]) << 8 | @as(u16, slice[0]); | ||||
|     return switch (id) { | ||||
|         0x3130 => "Nintendo", | ||||
|         else => null, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: Self) void { | ||||
|     self.alloc.free(self.buf); | ||||
|     self.backup.deinit(); | ||||
| } | ||||
|  | ||||
| pub fn read(self: *const Self, comptime T: type, address: u32) T { | ||||
|     const addr = address & 0x1FF_FFFF; | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => (@as(T, self.get(addr + 3)) << 24) | (@as(T, self.get(addr + 2)) << 16) | (@as(T, self.get(addr + 1)) << 8) | (@as(T, self.get(addr))), | ||||
|         u16 => (@as(T, self.get(addr + 1)) << 8) | @as(T, self.get(addr)), | ||||
|         u8 => self.get(addr), | ||||
|         else => @compileError("GamePak: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| fn get(self: *const Self, i: u32) u8 { | ||||
|     @setRuntimeSafety(false); | ||||
|  | ||||
|     if (i >= self.buf.len) { | ||||
|         const lhs = i >> 1 & 0xFFFF; | ||||
|         return @truncate(u8, lhs >> 8 * @truncate(u5, i & 1)); | ||||
|     } | ||||
|  | ||||
|     return self.buf[i]; | ||||
| } | ||||
|  | ||||
| test "OOB Access" { | ||||
|     const title = .{ 'H', 'E', 'L', 'L', 'O', ' ', 'W', 'O', 'R', 'L', 'D', '!' }; | ||||
|     const alloc = std.testing.allocator; | ||||
|     const pak = Self{ | ||||
|         .buf = &.{}, | ||||
|         .alloc = alloc, | ||||
|         .title = title, | ||||
|         .backup = try Backup.init(alloc, .None, title, null), | ||||
|     }; | ||||
|  | ||||
|     std.debug.assert(pak.get(0) == 0x00); // 0x0000 | ||||
|     std.debug.assert(pak.get(1) == 0x00); | ||||
|  | ||||
|     std.debug.assert(pak.get(2) == 0x01); // 0x0001 | ||||
|     std.debug.assert(pak.get(3) == 0x00); | ||||
|  | ||||
|     std.debug.assert(pak.get(4) == 0x02); // 0x0002 | ||||
|     std.debug.assert(pak.get(5) == 0x00); | ||||
| } | ||||
							
								
								
									
										52
									
								
								src/bus/Iwram.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/bus/Iwram.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
| const iwram_size = 0x8000; | ||||
| const Self = @This(); | ||||
|  | ||||
| buf: []u8, | ||||
| alloc: Allocator, | ||||
|  | ||||
| pub fn init(alloc: Allocator) !Self { | ||||
|     const buf = try alloc.alloc(u8, iwram_size); | ||||
|     std.mem.set(u8, buf, 0); | ||||
|  | ||||
|     return Self{ | ||||
|         .buf = buf, | ||||
|         .alloc = alloc, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: Self) void { | ||||
|     self.alloc.free(self.buf); | ||||
| } | ||||
|  | ||||
| pub fn read(self: *const Self, comptime T: type, address: usize) T { | ||||
|     const addr = address & 0x7FFF; | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => (@as(u32, self.buf[addr + 3]) << 24) | (@as(u32, self.buf[addr + 2]) << 16) | (@as(u32, self.buf[addr + 1]) << 8) | (@as(u32, self.buf[addr])), | ||||
|         u16 => (@as(u16, self.buf[addr + 1]) << 8) | @as(u16, self.buf[addr]), | ||||
|         u8 => self.buf[addr], | ||||
|         else => @compileError("IWRAM: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(self: *const Self, comptime T: type, address: usize, value: T) void { | ||||
|     const addr = address & 0x7FFF; | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => { | ||||
|             self.buf[addr + 3] = @truncate(u8, value >> 24); | ||||
|             self.buf[addr + 2] = @truncate(u8, value >> 16); | ||||
|             self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|             self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|         }, | ||||
|         u16 => { | ||||
|             self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|             self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|         }, | ||||
|         u8 => self.buf[addr] = value, | ||||
|         else => @compileError("IWRAM: Unsupported write width"), | ||||
|     }; | ||||
| } | ||||
							
								
								
									
										277
									
								
								src/bus/backup.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										277
									
								
								src/bus/backup.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,277 @@ | ||||
| const std = @import("std"); | ||||
| const Allocator = std.mem.Allocator; | ||||
| const log = std.log.scoped(.Backup); | ||||
|  | ||||
| const correctTitle = @import("../util.zig").correctTitle; | ||||
| const safeTitle = @import("../util.zig").safeTitle; | ||||
|  | ||||
| const backup_kinds = [5]Needle{ | ||||
|     .{ .str = "EEPROM_V", .kind = .Eeprom }, | ||||
|     .{ .str = "SRAM_V", .kind = .Sram }, | ||||
|     .{ .str = "FLASH_V", .kind = .Flash }, | ||||
|     .{ .str = "FLASH512_V", .kind = .Flash }, | ||||
|     .{ .str = "FLASH1M_V", .kind = .Flash1M }, | ||||
| }; | ||||
|  | ||||
| pub const Backup = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     buf: []u8, | ||||
|     alloc: Allocator, | ||||
|     kind: BackupKind, | ||||
|  | ||||
|     title: [12]u8, | ||||
|     save_path: ?[]const u8, | ||||
|  | ||||
|     // TODO: Implement EEPROM | ||||
|     flash: Flash, | ||||
|  | ||||
|     pub fn init(alloc: Allocator, kind: BackupKind, title: [12]u8, path: ?[]const u8) !Self { | ||||
|         log.info("Kind: {}", .{kind}); | ||||
|  | ||||
|         const buf_size: usize = switch (kind) { | ||||
|             .Sram => 0x8000, // 32K | ||||
|             .Flash => 0x10000, // 64K | ||||
|             .Flash1M => 0x20000, // 128K | ||||
|             .Eeprom => 0x2000, // FIXME: We assume 8K here | ||||
|             .None => 0, | ||||
|         }; | ||||
|  | ||||
|         const buf = try alloc.alloc(u8, buf_size); | ||||
|         std.mem.set(u8, buf, 0xFF); | ||||
|  | ||||
|         var backup = Self{ | ||||
|             .buf = buf, | ||||
|             .alloc = alloc, | ||||
|             .kind = kind, | ||||
|             .title = title, | ||||
|             .save_path = path, | ||||
|             .flash = Flash.init(), | ||||
|         }; | ||||
|  | ||||
|         if (backup.save_path) |p| backup.loadSaveFromDisk(p) catch |e| log.err("Failed to load save: {}", .{e}); | ||||
|         return backup; | ||||
|     } | ||||
|  | ||||
|     pub fn guessKind(rom: []const u8) ?BackupKind { | ||||
|         for (backup_kinds) |needle| { | ||||
|             const needle_len = needle.str.len; | ||||
|  | ||||
|             var i: usize = 0; | ||||
|             while ((i + needle_len) < rom.len) : (i += 1) { | ||||
|                 if (std.mem.eql(u8, needle.str, rom[i..][0..needle_len])) return needle.kind; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     pub fn deinit(self: Self) void { | ||||
|         if (self.save_path) |path| self.writeSaveToDisk(path) catch |e| log.err("Failed to write save: {}", .{e}); | ||||
|         self.alloc.free(self.buf); | ||||
|     } | ||||
|  | ||||
|     fn loadSaveFromDisk(self: *Self, path: []const u8) !void { | ||||
|         const file_path = try self.getSaveFilePath(path); | ||||
|         defer self.alloc.free(file_path); | ||||
|  | ||||
|         const file: std.fs.File = try std.fs.openFileAbsolute(file_path, .{}); | ||||
|         const file_buf = try file.readToEndAlloc(self.alloc, try file.getEndPos()); | ||||
|         defer self.alloc.free(file_buf); | ||||
|  | ||||
|         switch (self.kind) { | ||||
|             .Sram, .Flash, .Flash1M => { | ||||
|                 if (self.buf.len == file_buf.len) { | ||||
|                     std.mem.copy(u8, self.buf, file_buf); | ||||
|                     return log.info("Loaded Save from {s}", .{file_path}); | ||||
|                 } | ||||
|  | ||||
|                 log.err("{s} is {} bytes, but we expected {} bytes", .{ file_path, file_buf.len, self.buf.len }); | ||||
|             }, | ||||
|             else => return SaveError.UnsupportedBackupKind, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn getSaveFilePath(self: *const Self, path: []const u8) ![]const u8 { | ||||
|         const filename = try self.getSaveFilename(); | ||||
|         defer self.alloc.free(filename); | ||||
|  | ||||
|         return try std.fs.path.join(self.alloc, &[_][]const u8{ path, filename }); | ||||
|     } | ||||
|  | ||||
|     fn getSaveFilename(self: *const Self) ![]const u8 { | ||||
|         const title = correctTitle(safeTitle(self.title)); | ||||
|         return try std.mem.concat(self.alloc, u8, &[_][]const u8{ title, ".sav" }); | ||||
|     } | ||||
|  | ||||
|     fn writeSaveToDisk(self: Self, path: []const u8) !void { | ||||
|         const file_path = try self.getSaveFilePath(path); | ||||
|         defer self.alloc.free(file_path); | ||||
|  | ||||
|         switch (self.kind) { | ||||
|             .Sram, .Flash, .Flash1M => { | ||||
|                 const file = try std.fs.createFileAbsolute(file_path, .{}); | ||||
|                 defer file.close(); | ||||
|  | ||||
|                 try file.writeAll(self.buf); | ||||
|                 log.info("Wrote Save to {s}", .{file_path}); | ||||
|             }, | ||||
|             else => return SaveError.UnsupportedBackupKind, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn read(self: *const Self, address: usize) u8 { | ||||
|         const addr = address & 0xFFFF; | ||||
|  | ||||
|         switch (self.kind) { | ||||
|             .Flash => { | ||||
|                 switch (addr) { | ||||
|                     0x0000 => if (self.flash.id_mode) return 0x32, // Panasonic manufacturer ID | ||||
|                     0x0001 => if (self.flash.id_mode) return 0x1B, // Panasonic device ID | ||||
|                     else => {}, | ||||
|                 } | ||||
|  | ||||
|                 return self.flash.read(self.buf, addr); | ||||
|             }, | ||||
|             .Flash1M => { | ||||
|                 switch (addr) { | ||||
|                     0x0000 => if (self.flash.id_mode) return 0x62, // Sanyo manufacturer ID | ||||
|                     0x0001 => if (self.flash.id_mode) return 0x13, // Sanyo device ID | ||||
|                     else => {}, | ||||
|                 } | ||||
|  | ||||
|                 return self.flash.read(self.buf, addr); | ||||
|             }, | ||||
|             .Eeprom => return self.buf[addr], | ||||
|             .Sram => return self.buf[addr & 0x7FFF], // 32K SRAM chip is mirrored | ||||
|             .None => return 0xFF, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn write(self: *Self, address: usize, byte: u8) void { | ||||
|         const addr = address & 0xFFFF; | ||||
|  | ||||
|         switch (self.kind) { | ||||
|             .Flash, .Flash1M => { | ||||
|                 if (self.flash.prep_write) return self.flash.write(self.buf, addr, byte); | ||||
|                 if (self.flash.shouldEraseSector(addr, byte)) return self.flash.eraseSector(self.buf, addr); | ||||
|  | ||||
|                 switch (addr) { | ||||
|                     0x0000 => if (self.kind == .Flash1M and self.flash.set_bank) { | ||||
|                         self.flash.bank = @truncate(u1, byte); | ||||
|                     }, | ||||
|                     0x5555 => { | ||||
|                         if (self.flash.state == .Command) { | ||||
|                             self.flash.handleCommand(self.buf, byte); | ||||
|                         } else if (byte == 0xAA and self.flash.state == .Ready) { | ||||
|                             self.flash.state = .Set; | ||||
|                         } else if (byte == 0xF0) { | ||||
|                             self.flash.state = .Ready; | ||||
|                         } | ||||
|                     }, | ||||
|                     0x2AAA => if (byte == 0x55 and self.flash.state == .Set) { | ||||
|                         self.flash.state = .Command; | ||||
|                     }, | ||||
|                     else => {}, | ||||
|                 } | ||||
|             }, | ||||
|             .Eeprom => self.buf[addr] = byte, | ||||
|             .Sram => self.buf[addr & 0x7FFF] = byte, | ||||
|             .None => {}, | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const BackupKind = enum { | ||||
|     Eeprom, | ||||
|     Sram, | ||||
|     Flash, | ||||
|     Flash1M, | ||||
|     None, | ||||
| }; | ||||
|  | ||||
| const Needle = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     str: []const u8, | ||||
|     kind: BackupKind, | ||||
|  | ||||
|     fn init(str: []const u8, kind: BackupKind) Self { | ||||
|         return .{ | ||||
|             .str = str, | ||||
|             .kind = kind, | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const SaveError = error{ | ||||
|     UnsupportedBackupKind, | ||||
| }; | ||||
|  | ||||
| const Flash = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     state: FlashState, | ||||
|  | ||||
|     id_mode: bool, | ||||
|     set_bank: bool, | ||||
|     prep_erase: bool, | ||||
|     prep_write: bool, | ||||
|  | ||||
|     bank: u1, | ||||
|  | ||||
|     fn init() Self { | ||||
|         return .{ | ||||
|             .state = .Ready, | ||||
|             .id_mode = false, | ||||
|             .set_bank = false, | ||||
|             .prep_erase = false, | ||||
|             .prep_write = false, | ||||
|             .bank = 0, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn handleCommand(self: *Self, buf: []u8, byte: u8) void { | ||||
|         switch (byte) { | ||||
|             0x90 => self.id_mode = true, | ||||
|             0xF0 => self.id_mode = false, | ||||
|             0xB0 => self.set_bank = true, | ||||
|             0x80 => self.prep_erase = true, | ||||
|             0x10 => { | ||||
|                 std.mem.set(u8, buf, 0xFF); | ||||
|                 self.prep_erase = false; | ||||
|             }, | ||||
|             0xA0 => self.prep_write = true, | ||||
|             else => std.debug.panic("Unhandled Flash Command: 0x{X:0>2}", .{byte}), | ||||
|         } | ||||
|  | ||||
|         self.state = .Ready; | ||||
|     } | ||||
|  | ||||
|     fn shouldEraseSector(self: *const Self, idx: usize, byte: u8) bool { | ||||
|         return self.prep_erase and idx & 0xFFF == 0x000 and byte == 0x30; | ||||
|     } | ||||
|  | ||||
|     fn write(self: *Self, buf: []u8, idx: usize, byte: u8) void { | ||||
|         buf[idx + if (self.bank == 1) 0x1000 else @as(usize, 0)] = byte; | ||||
|         self.prep_write = false; | ||||
|     } | ||||
|  | ||||
|     fn read(self: *const Self, buf: []u8, idx: usize) u8 { | ||||
|         return buf[idx + if (self.bank == 1) 0x1000 else @as(usize, 0)]; | ||||
|     } | ||||
|  | ||||
|     fn eraseSector(self: *Self, buf: []u8, idx: usize) void { | ||||
|         const start = (idx & 0xF000) + if (self.bank == 1) 0x1000 else @as(usize, 0); | ||||
|  | ||||
|         std.mem.set(u8, buf[start..][0..0x1000], 0xFF); | ||||
|         self.prep_erase = false; | ||||
|         self.state = .Ready; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const FlashState = enum { | ||||
|     Ready, | ||||
|     Set, | ||||
|     Command, | ||||
| }; | ||||
							
								
								
									
										256
									
								
								src/bus/dma.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										256
									
								
								src/bus/dma.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,256 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const DmaControl = @import("io.zig").DmaControl; | ||||
| const Bus = @import("../Bus.zig"); | ||||
|  | ||||
| const log = std.log.scoped(.DmaTransfer); | ||||
|  | ||||
| pub const DmaControllers = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     _0: DmaController(0), | ||||
|     _1: DmaController(1), | ||||
|     _2: DmaController(2), | ||||
|     _3: DmaController(3), | ||||
|  | ||||
|     pub fn init() Self { | ||||
|         return .{ | ||||
|             ._0 = DmaController(0).init(), | ||||
|             ._1 = DmaController(1).init(), | ||||
|             ._2 = DmaController(2).init(), | ||||
|             ._3 = DmaController(3).init(), | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| /// Function that creates a DMAController. Determines unique DMA Controller behaiour at compile-time | ||||
| fn DmaController(comptime id: u2) type { | ||||
|     return struct { | ||||
|         const Self = @This(); | ||||
|  | ||||
|         const sad_mask: u32 = if (id == 0) 0x07FF_FFFF else 0x0FFF_FFFF; | ||||
|         const dad_mask: u32 = if (id != 3) 0x07FF_FFFF else 0x0FFF_FFFF; | ||||
|  | ||||
|         /// Determines whether DMAController is for DMA0, DMA1, DMA2 or DMA3 | ||||
|         /// Note: Determined at comptime | ||||
|         id: u2, | ||||
|         /// Write-only. The first address in a DMA transfer. (DMASAD) | ||||
|         /// Note: use writeSrc instead of manipulating src_addr directly | ||||
|         sad: u32, | ||||
|         /// Write-only. The final address in a DMA transffer. (DMADAD) | ||||
|         /// Note: Use writeDst instead of manipulatig dst_addr directly | ||||
|         dad: u32, | ||||
|         /// Write-only. The Word Count for the DMA Transfer (DMACNT_L) | ||||
|         word_count: if (id == 3) u16 else u14, | ||||
|         /// Read / Write. DMACNT_H | ||||
|         /// Note: Use writeControl instead of manipulating cnt directly. | ||||
|         cnt: DmaControl, | ||||
|  | ||||
|         /// Internal. Currrent Source Address | ||||
|         _sad: u32, | ||||
|         /// Internal. Current Destination Address | ||||
|         _dad: u32, | ||||
|         /// Internal. Word Count | ||||
|         _word_count: if (id == 3) u16 else u14, | ||||
|  | ||||
|         // Internal. FIFO Word Count | ||||
|         _fifo_word_count: u8, | ||||
|  | ||||
|         /// Some DMA Transfers are enabled during Hblank / VBlank and / or | ||||
|         /// have delays. Thefore bit 15 of DMACNT isn't actually something | ||||
|         /// we can use to control when we do or do not execute a step in a DMA Transfer | ||||
|         active: bool, | ||||
|  | ||||
|         pub fn init() Self { | ||||
|             return .{ | ||||
|                 .id = id, | ||||
|                 .sad = 0, | ||||
|                 .dad = 0, | ||||
|                 .word_count = 0, | ||||
|                 .cnt = .{ .raw = 0x000 }, | ||||
|  | ||||
|                 // Internals | ||||
|                 ._sad = 0, | ||||
|                 ._dad = 0, | ||||
|                 ._word_count = 0, | ||||
|                 ._fifo_word_count = 4, | ||||
|                 .active = false, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         pub fn writeSad(self: *Self, addr: u32) void { | ||||
|             self.sad = addr & sad_mask; | ||||
|         } | ||||
|  | ||||
|         pub fn writeDad(self: *Self, addr: u32) void { | ||||
|             self.dad = addr & dad_mask; | ||||
|         } | ||||
|  | ||||
|         pub fn writeWordCount(self: *Self, halfword: u16) void { | ||||
|             self.word_count = @truncate(@TypeOf(self.word_count), halfword); | ||||
|         } | ||||
|  | ||||
|         pub fn writeCntHigh(self: *Self, halfword: u16) void { | ||||
|             const new = DmaControl{ .raw = halfword }; | ||||
|  | ||||
|             if (!self.cnt.enabled.read() and new.enabled.read()) { | ||||
|                 // Reload Internals on Rising Edge. | ||||
|                 self._sad = self.sad; | ||||
|                 self._dad = self.dad; | ||||
|                 self._word_count = if (self.word_count == 0) std.math.maxInt(@TypeOf(self._word_count)) else self.word_count; | ||||
|  | ||||
|                 // Only a Start Timing of 00 has a DMA Transfer immediately begin | ||||
|                 self.active = new.start_timing.read() == 0b00; | ||||
|             } | ||||
|  | ||||
|             self.cnt.raw = halfword; | ||||
|         } | ||||
|  | ||||
|         pub fn writeCnt(self: *Self, word: u32) void { | ||||
|             self.word_count = @truncate(@TypeOf(self.word_count), word); | ||||
|             self.writeCntHigh(@truncate(u16, word >> 16)); | ||||
|         } | ||||
|  | ||||
|         pub inline fn check(self: *Self, bus: *Bus) bool { | ||||
|             if (!self.active) return false; // FIXME: Check CNT register? | ||||
|  | ||||
|             self.step(bus); | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         pub fn step(self: *Self, bus: *Bus) bool { | ||||
|             if (!self.active) return false; | ||||
|  | ||||
|             const sad_adj = std.meta.intToEnum(Adjustment, self.cnt.sad_adj.read()) catch unreachable; | ||||
|             const dad_adj = std.meta.intToEnum(Adjustment, self.cnt.dad_adj.read()) catch unreachable; | ||||
|             const is_fifo = (self.id == 1 or self.id == 2) and self.cnt.start_timing.read() == 0b11; | ||||
|  | ||||
|             // // if (is_fifo) { | ||||
|             // //     const offset = @sizeOf(u32); | ||||
|             // //     bus.write(u32, self._dad, bus.read(u32, self._sad)); | ||||
|  | ||||
|             // //     // TODO: Deduplicate | ||||
|             // //     switch (sad_adj) { | ||||
|             // //         .Increment => self._sad +%= offset, | ||||
|             // //         .Decrement => self._sad -%= offset, | ||||
|             // //         .Fixed => {}, | ||||
|  | ||||
|             // //         // TODO: Figure out correct behaviour on Illegal Source Addr Control Type | ||||
|             // //         .IncrementReload => std.debug.panic("panic(DmaTransfer): {} is an illegal src addr adjustment type", .{sad_adj}), | ||||
|             // //     } | ||||
|  | ||||
|             // //     self._fifo_word_count -= 1; | ||||
|  | ||||
|             // //     if (self._fifo_word_count == 0) { | ||||
|             // //         self._fifo_word_count = 4; | ||||
|             // //         self.active = false; | ||||
|             // //     } | ||||
|  | ||||
|             // //     return true; | ||||
|             // // } | ||||
|  | ||||
|             const transfer_type = self.cnt.transfer_type.read() or is_fifo; | ||||
|             const offset: u32 = if (transfer_type) @sizeOf(u32) else @sizeOf(u16); | ||||
|  | ||||
|             if (transfer_type) { | ||||
|                 bus.write(u32, self._dad, bus.read(u32, self._sad)); | ||||
|             } else { | ||||
|                 bus.write(u16, self._dad, bus.read(u16, self._sad)); | ||||
|             } | ||||
|  | ||||
|             switch (sad_adj) { | ||||
|                 .Increment => self._sad +%= offset, | ||||
|                 .Decrement => self._sad -%= offset, | ||||
|                 .Fixed => {}, | ||||
|  | ||||
|                 // TODO: Figure out correct behaviour on Illegal Source Addr Control Type | ||||
|                 .IncrementReload => std.debug.panic("panic(DmaTransfer): {} is an illegal src addr adjustment type", .{sad_adj}), | ||||
|             } | ||||
|  | ||||
|             if (!is_fifo) { | ||||
|                 switch (dad_adj) { | ||||
|                     .Increment, .IncrementReload => self._dad +%= offset, | ||||
|                     .Decrement => self._dad -%= offset, | ||||
|                     .Fixed => {}, | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             self._word_count -= 1; | ||||
|  | ||||
|             if (self._word_count == 0) { | ||||
|                 if (!self.cnt.repeat.read()) { | ||||
|                     // If we're not repeating, Fire the IRQs and disable the DMA | ||||
|                     if (self.cnt.irq.read()) { | ||||
|                         switch (id) { | ||||
|                             0 => bus.io.irq.dma0.set(), | ||||
|                             1 => bus.io.irq.dma0.set(), | ||||
|                             2 => bus.io.irq.dma0.set(), | ||||
|                             3 => bus.io.irq.dma0.set(), | ||||
|                         } | ||||
|                     } | ||||
|                     self.cnt.enabled.unset(); | ||||
|                 } | ||||
|  | ||||
|                 // We want to disable our internal enabled flag regardless of repeat | ||||
|                 // because we only want to step A DMA that repeats during it's specific | ||||
|                 // timing window | ||||
|                 self.active = false; | ||||
|             } | ||||
|  | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         pub fn isBlocking(self: *const Self) bool { | ||||
|             // A DMA Transfer is Blocking if it is Immediate | ||||
|             return self.cnt.start_timing.read() == 0b00; | ||||
|         } | ||||
|  | ||||
|         pub fn pollBlankingDma(self: *Self, comptime kind: DmaKind) void { | ||||
|             if (self.active) return; | ||||
|  | ||||
|             switch (kind) { | ||||
|                 .HBlank => self.active = self.cnt.enabled.read() and self.cnt.start_timing.read() == 0b10, | ||||
|                 .VBlank => self.active = self.cnt.enabled.read() and self.cnt.start_timing.read() == 0b01, | ||||
|                 .Immediate, .Special => {}, | ||||
|             } | ||||
|  | ||||
|             if (self.cnt.repeat.read() and self.active) { | ||||
|                 self._word_count = if (self.word_count == 0) std.math.maxInt(@TypeOf(self._word_count)) else self.word_count; | ||||
|  | ||||
|                 const dad_adj = std.meta.intToEnum(Adjustment, self.cnt.dad_adj.read()) catch unreachable; | ||||
|                 if (dad_adj == .IncrementReload) self._dad = self.dad; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         pub fn enableSoundDma(self: *Self, fifo_addr: u32) void { | ||||
|             comptime std.debug.assert(id == 1 or id == 2); | ||||
|  | ||||
|             if (self.cnt.enabled.read() and self.cnt.start_timing.read() == 0b11 and self.dad == fifo_addr) { | ||||
|                 self.active = true; | ||||
|                 self._word_count = 4; | ||||
|                 self.cnt.repeat.set(); | ||||
|             } | ||||
|         } | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn pollBlankingDma(bus: *Bus, comptime kind: DmaKind) void { | ||||
|     bus.dma._0.pollBlankingDma(kind); | ||||
|     bus.dma._1.pollBlankingDma(kind); | ||||
|     bus.dma._2.pollBlankingDma(kind); | ||||
|     bus.dma._3.pollBlankingDma(kind); | ||||
| } | ||||
|  | ||||
| const Adjustment = enum(u2) { | ||||
|     Increment = 0, | ||||
|     Decrement = 1, | ||||
|     Fixed = 2, | ||||
|     IncrementReload = 3, | ||||
| }; | ||||
|  | ||||
| const DmaKind = enum(u2) { | ||||
|     Immediate = 0, | ||||
|     HBlank, | ||||
|     VBlank, | ||||
|     Special, | ||||
| }; | ||||
							
								
								
									
										634
									
								
								src/bus/io.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										634
									
								
								src/bus/io.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,634 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Bit = @import("bitfield").Bit; | ||||
| const Bitfield = @import("bitfield").Bitfield; | ||||
| const Bus = @import("../Bus.zig"); | ||||
| const DmaController = @import("dma.zig").DmaController; | ||||
| const Scheduler = @import("../scheduler.zig").Scheduler; | ||||
|  | ||||
| const panic_on_und_io: bool = false; | ||||
|  | ||||
| const log = std.log.scoped(.@"I/O"); | ||||
|  | ||||
| pub const Io = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     /// Read / Write | ||||
|     ime: bool, | ||||
|     ie: InterruptEnable, | ||||
|     irq: InterruptRequest, | ||||
|     postflg: PostFlag, | ||||
|     haltcnt: HaltControl, | ||||
|     keyinput: KeyInput, | ||||
|  | ||||
|     pub fn init() Self { | ||||
|         return .{ | ||||
|             .ime = false, | ||||
|             .ie = .{ .raw = 0x0000 }, | ||||
|             .irq = .{ .raw = 0x0000 }, | ||||
|             .keyinput = .{ .raw = 0x03FF }, | ||||
|             .postflg = .FirstBoot, | ||||
|             .haltcnt = .Execute, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn setIrqs(self: *Io, word: u32) void { | ||||
|         self.ie.raw = @truncate(u16, word); | ||||
|         self.irq.raw &= ~@truncate(u16, word >> 16); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| pub fn read(bus: *const Bus, comptime T: type, address: u32) T { | ||||
|     return switch (T) { | ||||
|         u32 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000 => bus.ppu.dispcnt.raw, | ||||
|             0x0400_0004 => @as(T, bus.ppu.vcount.raw) << 16 | bus.ppu.dispstat.raw, | ||||
|             0x0400_0006 => @as(T, bus.ppu.bg[0].cnt.raw) << 16 | bus.ppu.vcount.raw, | ||||
|  | ||||
|             // DMA Transfers | ||||
|             0x0400_00B8 => @as(T, bus.dma._0.cnt.raw) << 16, | ||||
|             0x0400_00C4 => @as(T, bus.dma._1.cnt.raw) << 16, | ||||
|             0x0400_00D0 => @as(T, bus.dma._1.cnt.raw) << 16, | ||||
|             0x0400_00DC => @as(T, bus.dma._3.cnt.raw) << 16, | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100 => @as(T, bus.tim._0.cnt.raw) << 16 | bus.tim._0.counter(), | ||||
|             0x0400_0104 => @as(T, bus.tim._1.cnt.raw) << 16 | bus.tim._1.counter(), | ||||
|             0x0400_0108 => @as(T, bus.tim._2.cnt.raw) << 16 | bus.tim._2.counter(), | ||||
|             0x0400_010C => @as(T, bus.tim._3.cnt.raw) << 16 | bus.tim._3.counter(), | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200 => @as(T, bus.io.irq.raw) << 16 | bus.io.ie.raw, | ||||
|             0x0400_0208 => @boolToInt(bus.io.ime), | ||||
|             else => undRead("Tried to read {} from 0x{X:0>8}", .{ T, address }), | ||||
|         }, | ||||
|         u16 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000 => bus.ppu.dispcnt.raw, | ||||
|             0x0400_0004 => bus.ppu.dispstat.raw, | ||||
|             0x0400_0006 => bus.ppu.vcount.raw, | ||||
|             0x0400_0008 => bus.ppu.bg[0].cnt.raw, | ||||
|             0x0400_000A => bus.ppu.bg[1].cnt.raw, | ||||
|             0x0400_000C => bus.ppu.bg[2].cnt.raw, | ||||
|             0x0400_000E => bus.ppu.bg[3].cnt.raw, | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0088 => bus.apu.bias.raw, | ||||
|  | ||||
|             // DMA Transfers | ||||
|             0x0400_00BA => bus.dma._0.cnt.raw, | ||||
|             0x0400_00C6 => bus.dma._1.cnt.raw, | ||||
|             0x0400_00D2 => bus.dma._2.cnt.raw, | ||||
|             0x0400_00DE => bus.dma._3.cnt.raw, | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100 => bus.tim._0.counter(), | ||||
|             0x0400_0102 => bus.tim._0.cnt.raw, | ||||
|             0x0400_0104 => bus.tim._1.counter(), | ||||
|             0x0400_0106 => bus.tim._1.cnt.raw, | ||||
|             0x0400_0108 => bus.tim._2.counter(), | ||||
|             0x0400_010A => bus.tim._2.cnt.raw, | ||||
|             0x0400_010C => bus.tim._3.counter(), | ||||
|             0x0400_010E => bus.tim._3.cnt.raw, | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0128 => unimplementedRead("Read halfword from SIOCNT", .{}), | ||||
|  | ||||
|             // Keypad Input | ||||
|             0x0400_0130 => bus.io.keyinput.raw, | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200 => bus.io.ie.raw, | ||||
|             0x0400_0202 => bus.io.irq.raw, | ||||
|             0x0400_0204 => unimplementedRead("Read halfword from WAITCNT", .{}), | ||||
|             0x0400_0208 => @boolToInt(bus.io.ime), | ||||
|             else => undRead("Tried to read {} from 0x{X:0>8}", .{ T, address }), | ||||
|         }, | ||||
|         u8 => return switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000 => @truncate(T, bus.ppu.dispcnt.raw), | ||||
|             0x0400_0004 => @truncate(T, bus.ppu.dispstat.raw), | ||||
|             0x0400_0006 => @truncate(T, bus.ppu.vcount.raw), | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0060 => bus.apu.ch1.sweep.raw, | ||||
|             0x0400_0063 => bus.apu.ch1.envelope.raw, | ||||
|             0x0400_0069 => bus.apu.ch2.envelope.raw, | ||||
|             0x0400_0073 => bus.apu.ch3.vol.raw, | ||||
|             0x0400_0079 => bus.apu.ch4.envelope.raw, | ||||
|             0x0400_007C => bus.apu.ch4.poly.raw, | ||||
|             0x0400_0081 => @truncate(T, bus.apu.ch_vol_cnt.raw >> 8), | ||||
|             0x0400_0089 => @truncate(T, bus.apu.bias.raw >> 8), | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0128 => unimplementedRead("Read (low) byte from SIOCNT", .{}), | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200 => @truncate(T, bus.io.ie.raw), | ||||
|             0x0400_0300 => @enumToInt(bus.io.postflg), | ||||
|             else => undRead("Tried to read byte from 0x{X:0>8}", .{address}), | ||||
|         }, | ||||
|         else => @compileError("I/O: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(bus: *Bus, comptime T: type, address: u32, value: T) void { | ||||
|     return switch (T) { | ||||
|         u32 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000 => bus.ppu.dispcnt.raw = @truncate(u16, value), | ||||
|             0x0400_0004 => { | ||||
|                 bus.ppu.dispstat.raw = @truncate(u16, value); | ||||
|                 bus.ppu.vcount.raw = @truncate(u16, value >> 16); | ||||
|             }, | ||||
|             0x0400_0008 => bus.ppu.setAdjCnts(0, value), | ||||
|             0x0400_000C => bus.ppu.setAdjCnts(2, value), | ||||
|             0x0400_0010 => bus.ppu.setBgOffsets(0, value), | ||||
|             0x0400_0014 => bus.ppu.setBgOffsets(1, value), | ||||
|             0x0400_0018 => bus.ppu.setBgOffsets(2, value), | ||||
|             0x0400_001C => bus.ppu.setBgOffsets(3, value), | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0080 => { | ||||
|                 bus.apu.ch_vol_cnt.raw = @truncate(u16, value); | ||||
|                 bus.apu.dma_cnt.raw = @truncate(u16, value >> 16); | ||||
|             }, | ||||
|             0x0400_00A0 => bus.apu.chA.push(value), | ||||
|             0x0400_00A4 => bus.apu.chB.push(value), | ||||
|  | ||||
|             // DMA Transfers | ||||
|             0x0400_00B0 => bus.dma._0.writeSad(value), | ||||
|             0x0400_00B4 => bus.dma._0.writeDad(value), | ||||
|             0x0400_00B8 => bus.dma._0.writeCnt(value), | ||||
|             0x0400_00BC => bus.dma._1.writeSad(value), | ||||
|             0x0400_00C0 => bus.dma._1.writeDad(value), | ||||
|             0x0400_00C4 => bus.dma._1.writeCnt(value), | ||||
|             0x0400_00C8 => bus.dma._2.writeSad(value), | ||||
|             0x0400_00CC => bus.dma._2.writeDad(value), | ||||
|             0x0400_00D0 => bus.dma._2.writeCnt(value), | ||||
|             0x0400_00D4 => bus.dma._3.writeSad(value), | ||||
|             0x0400_00D8 => bus.dma._3.writeDad(value), | ||||
|             0x0400_00DC => bus.dma._3.writeCnt(value), | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100 => bus.tim._0.writeCnt(value), | ||||
|             0x0400_0104 => bus.tim._1.writeCnt(value), | ||||
|             0x0400_0108 => bus.tim._2.writeCnt(value), | ||||
|             0x0400_010C => bus.tim._3.writeCnt(value), | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0120 => log.warn("Wrote 0x{X:0>8} to SIODATA32", .{value}), | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200 => bus.io.setIrqs(value), | ||||
|             0x0400_0204 => log.warn("Wrote 0x{X:0>8} to WAITCNT", .{value}), | ||||
|             0x0400_0208 => bus.io.ime = value & 1 == 1, | ||||
|             else => undWrite("Tried to write {} 0x{X:0>8} to 0x{X:0>8}", .{ T, value, address }), | ||||
|         }, | ||||
|         u16 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000 => bus.ppu.dispcnt.raw = value, | ||||
|             0x0400_0004 => bus.ppu.dispstat.raw = value, | ||||
|             0x0400_0006 => {}, // vcount is read-only | ||||
|             0x0400_0008 => bus.ppu.bg[0].cnt.raw = value, | ||||
|             0x0400_000A => bus.ppu.bg[1].cnt.raw = value, | ||||
|             0x0400_000C => bus.ppu.bg[2].cnt.raw = value, | ||||
|             0x0400_000E => bus.ppu.bg[3].cnt.raw = value, | ||||
|             0x0400_0010 => bus.ppu.bg[0].hofs.raw = value, // TODO: Don't write out every HOFS / VOFS? | ||||
|             0x0400_0012 => bus.ppu.bg[0].vofs.raw = value, | ||||
|             0x0400_0014 => bus.ppu.bg[1].hofs.raw = value, | ||||
|             0x0400_0016 => bus.ppu.bg[1].vofs.raw = value, | ||||
|             0x0400_0018 => bus.ppu.bg[2].hofs.raw = value, | ||||
|             0x0400_001A => bus.ppu.bg[2].vofs.raw = value, | ||||
|             0x0400_001C => bus.ppu.bg[3].hofs.raw = value, | ||||
|             0x0400_001E => bus.ppu.bg[3].vofs.raw = value, | ||||
|             0x0400_0020 => log.warn("Wrote 0x{X:0>4} to BG2PA", .{value}), | ||||
|             0x0400_0022 => log.warn("Wrote 0x{X:0>4} to BG2PB", .{value}), | ||||
|             0x0400_0024 => log.warn("Wrote 0x{X:0>4} to BG2PC", .{value}), | ||||
|             0x0400_0026 => log.warn("Wrote 0x{X:0>4} to BG2PD", .{value}), | ||||
|             0x0400_0028 => log.warn("Wrote 0x{X:0>4} to BG2X_L", .{value}), | ||||
|             0x0400_002A => log.warn("Wrote 0x{X:0>4} to BG2X_H", .{value}), | ||||
|             0x0400_002C => log.warn("Wrote 0x{X:0>4} to BG2Y_L", .{value}), | ||||
|             0x0400_002E => log.warn("Wrote 0x{X:0>4} to BG2Y_H", .{value}), | ||||
|             0x0400_0030 => log.warn("Wrote 0x{X:0>4} to BG3PA", .{value}), | ||||
|             0x0400_0032 => log.warn("Wrote 0x{X:0>4} to BG3PB", .{value}), | ||||
|             0x0400_0034 => log.warn("Wrote 0x{X:0>4} to BG3PC", .{value}), | ||||
|             0x0400_0036 => log.warn("Wrote 0x{X:0>4} to BG3PD", .{value}), | ||||
|             0x0400_0038 => log.warn("Wrote 0x{X:0>4} to BG3X_L", .{value}), | ||||
|             0x0400_003A => log.warn("Wrote 0x{X:0>4} to BG3X_H", .{value}), | ||||
|             0x0400_003C => log.warn("Wrote 0x{X:0>4} to BG3Y_L", .{value}), | ||||
|             0x0400_003E => log.warn("Wrote 0x{X:0>4} to BG3Y_H", .{value}), | ||||
|             0x0400_0040 => log.warn("Wrote 0x{X:0>4} to WIN0H", .{value}), | ||||
|             0x0400_0042 => log.warn("Wrote 0x{X:0>4} to WIN1H", .{value}), | ||||
|             0x0400_0044 => log.warn("Wrote 0x{X:0>4} to WIN0V", .{value}), | ||||
|             0x0400_0046 => log.warn("Wrote 0x{X:0>4} to WIN1V", .{value}), | ||||
|             0x0400_0048 => log.warn("Wrote 0x{X:0>4} to WININ", .{value}), | ||||
|             0x0400_004A => log.warn("Wrote 0x{X:0>4} to WINOUT", .{value}), | ||||
|             0x0400_004C => log.warn("Wrote 0x{X:0>4} to MOSAIC", .{value}), | ||||
|             0x0400_0050 => log.warn("Wrote 0x{X:0>4} to BLDCNT", .{value}), | ||||
|             0x0400_0052 => log.warn("Wrote 0x{X:0>4} to BLDALPHA", .{value}), | ||||
|             0x0400_0054 => log.warn("Wrote 0x{X:0>4} to BLDY", .{value}), | ||||
|             0x0400_004E, 0x0400_0056 => {}, // Not used | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0080 => bus.apu.ch_vol_cnt.raw = value, | ||||
|             0x0400_0082 => bus.apu.setDmaCnt(value), | ||||
|             0x0400_0084 => bus.apu.setSoundCntX(value >> 7 & 1 == 1), | ||||
|             0x0400_0088 => bus.apu.bias.raw = value, | ||||
|             0x0400_0090...0x0400_009F => log.warn("Wrote 0x{X:0>4} to WAVE_RAM", .{value}), | ||||
|  | ||||
|             // Dma Transfers | ||||
|             0x0400_00B0 => bus.dma._0.writeSad(bus.dma._0.sad & 0xFFFF_0000 | value), | ||||
|             0x0400_00B2 => bus.dma._0.writeSad(bus.dma._0.sad & 0x0000_FFFF | (@as(u32, value) << 16)), | ||||
|             0x0400_00B4 => bus.dma._0.writeDad(bus.dma._0.dad & 0xFFFF_0000 | value), | ||||
|             0x0400_00B6 => bus.dma._0.writeDad(bus.dma._0.dad & 0x0000_FFFF | (@as(u32, value) << 16)), | ||||
|             0x0400_00B8 => bus.dma._0.writeWordCount(value), | ||||
|             0x0400_00BA => bus.dma._0.writeCntHigh(value), | ||||
|  | ||||
|             0x0400_00BC => bus.dma._1.writeSad(bus.dma._1.sad & 0xFFFF_0000 | value), | ||||
|             0x0400_00BE => bus.dma._1.writeSad(bus.dma._1.sad & 0x0000_FFFF | (@as(u32, value) << 16)), | ||||
|             0x0400_00C0 => bus.dma._1.writeDad(bus.dma._1.dad & 0xFFFF_0000 | value), | ||||
|             0x0400_00C2 => bus.dma._1.writeDad(bus.dma._1.dad & 0x0000_FFFF | (@as(u32, value) << 16)), | ||||
|             0x0400_00C4 => bus.dma._1.writeWordCount(value), | ||||
|             0x0400_00C6 => bus.dma._1.writeCntHigh(value), | ||||
|  | ||||
|             0x0400_00C8 => bus.dma._2.writeSad(bus.dma._2.sad & 0xFFFF_0000 | value), | ||||
|             0x0400_00CA => bus.dma._2.writeSad(bus.dma._2.sad & 0x0000_FFFF | (@as(u32, value) << 16)), | ||||
|             0x0400_00CC => bus.dma._2.writeDad(bus.dma._2.dad & 0xFFFF_0000 | value), | ||||
|             0x0400_00CE => bus.dma._2.writeDad(bus.dma._2.dad & 0x0000_FFFF | (@as(u32, value) << 16)), | ||||
|             0x0400_00D0 => bus.dma._2.writeWordCount(value), | ||||
|             0x0400_00D2 => bus.dma._2.writeCntHigh(value), | ||||
|  | ||||
|             0x0400_00D4 => bus.dma._3.writeSad(bus.dma._3.sad & 0xFFFF_0000 | value), | ||||
|             0x0400_00D6 => bus.dma._3.writeSad(bus.dma._3.sad & 0x0000_FFFF | (@as(u32, value) << 16)), | ||||
|             0x0400_00D8 => bus.dma._3.writeDad(bus.dma._3.dad & 0xFFFF_0000 | value), | ||||
|             0x0400_00DA => bus.dma._3.writeDad(bus.dma._3.dad & 0x0000_FFFF | (@as(u32, value) << 16)), | ||||
|             0x0400_00DC => bus.dma._3.writeWordCount(value), | ||||
|             0x0400_00DE => bus.dma._3.writeCntHigh(value), | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100 => bus.tim._0.writeCntLow(value), | ||||
|             0x0400_0102 => bus.tim._0.writeCntHigh(value), | ||||
|             0x0400_0104 => bus.tim._1.writeCntLow(value), | ||||
|             0x0400_0106 => bus.tim._1.writeCntHigh(value), | ||||
|             0x0400_0108 => bus.tim._2.writeCntLow(value), | ||||
|             0x0400_010A => bus.tim._2.writeCntHigh(value), | ||||
|             0x0400_010C => bus.tim._3.writeCntLow(value), | ||||
|             0x0400_010E => bus.tim._3.writeCntHigh(value), | ||||
|             0x0400_0110 => {}, // Not Used | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0120 => log.warn("Wrote 0x{X:0>4} to SIOMULTI0", .{value}), | ||||
|             0x0400_0122 => log.warn("Wrote 0x{X:0>4} to SIOMULTI1", .{value}), | ||||
|             0x0400_0124 => log.warn("Wrote 0x{X:0>4} to SIOMULTI2", .{value}), | ||||
|             0x0400_0126 => log.warn("Wrote 0x{X:0>4} to SIOMULTI3", .{value}), | ||||
|             0x0400_0128 => log.warn("Wrote 0x{X:0>4} to SIOCNT", .{value}), | ||||
|             0x0400_012A => log.warn("Wrote 0x{X:0>4} to SIOMLT_SEND", .{value}), | ||||
|  | ||||
|             // Keypad Input | ||||
|             0x0400_0130 => log.warn("Wrote 0x{X:0>4} to KEYINPUT. Ignored", .{value}), | ||||
|             0x0400_0132 => log.warn("Wrote 0x{X:0>4} to KEYCNT", .{value}), | ||||
|  | ||||
|             // Serial Communication 2 | ||||
|             0x0400_0134 => log.warn("Wrote 0x{X:0>4} to RCNT", .{value}), | ||||
|             0x0400_0140 => log.warn("Wrote 0x{X:0>4} to JOYCNT", .{value}), | ||||
|             0x0400_0158 => log.warn("Wrote 0x{X:0>4} to JOYSTAT", .{value}), | ||||
|             0x0400_0142, 0x0400_015A => {}, // Not Used | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200 => bus.io.ie.raw = value, | ||||
|             0x0400_0202 => bus.io.irq.raw &= ~value, | ||||
|             0x0400_0204 => log.warn("Wrote 0x{X:0>4} to WAITCNT", .{value}), | ||||
|             0x0400_0208 => bus.io.ime = value & 1 == 1, | ||||
|             0x0400_0206, 0x0400_020A => {}, // Not Used | ||||
|             else => undWrite("Tried to write {} 0x{X:0>4} to 0x{X:0>8}", .{ T, value, address }), | ||||
|         }, | ||||
|         u8 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0004 => bus.ppu.dispstat.raw = (bus.ppu.dispstat.raw & 0xFF00) | value, | ||||
|             0x0400_0005 => bus.ppu.dispstat.raw = (@as(u16, value) << 8) | (bus.ppu.dispstat.raw & 0xFF), | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0060 => bus.apu.ch1.sweep.raw = value, | ||||
|             0x0400_0062 => bus.apu.ch1.duty.raw = value, | ||||
|             0x0400_0063 => bus.apu.ch1.envelope.raw = value, | ||||
|             0x0400_0064 => bus.apu.ch1.setFreqLow(value), | ||||
|             0x0400_0065 => bus.apu.ch1.setFreqHigh(value), | ||||
|             0x0400_0068 => bus.apu.ch2.duty.raw = value, | ||||
|             0x0400_0069 => bus.apu.ch2.envelope.raw = value, | ||||
|             0x0400_006C => bus.apu.ch2.setFreqLow(value), | ||||
|             0x0400_006D => bus.apu.ch2.setFreqHigh(value), | ||||
|             0x0400_0070 => bus.apu.ch3.select.raw = value, | ||||
|             0x0400_0072 => bus.apu.ch3.length = value, | ||||
|             0x0400_0073 => bus.apu.ch3.vol.raw = value, | ||||
|             0x0400_0074 => bus.apu.ch3.setFreqLow(value), | ||||
|             0x0400_0075 => bus.apu.ch3.setFreqHigh(value), | ||||
|             0x0400_0078 => bus.apu.ch4.len = @truncate(u6, value), | ||||
|             0x0400_0079 => bus.apu.ch4.envelope.raw = value, | ||||
|             0x0400_007C => bus.apu.ch4.poly.raw = value, | ||||
|             0x0400_007D => bus.apu.ch4.cnt.raw = value, | ||||
|             0x0400_0080 => bus.apu.setSoundCntLLow(value), | ||||
|             0x0400_0081 => bus.apu.setSoundCntLHigh(value), | ||||
|             0x0400_0084 => bus.apu.setSoundCntX(value >> 7 & 1 == 1), | ||||
|             0x0400_0089 => bus.apu.setBiasHigh(value), | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0128 => log.warn("Wrote 0x{X:0>2} to SIOCNT (low)", .{value}), | ||||
|  | ||||
|             // Serial Communication 2 | ||||
|             0x0400_0140 => log.warn("Wrote 0x{X:0>2} to JOYCNT (low)", .{value}), | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0208 => bus.io.ime = value & 1 == 1, | ||||
|             0x0400_0301 => bus.io.haltcnt = if (value >> 7 & 1 == 0) .Halt else std.debug.panic("TODO: Implement STOP", .{}), | ||||
|             else => undWrite("Tried to write 0x{X:0>2} to 0x{X:0>8}", .{ value, address }), | ||||
|         }, | ||||
|         else => @compileError("I/O: Unsupported write width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| fn undRead(comptime format: []const u8, args: anytype) u8 { | ||||
|     if (panic_on_und_io) std.debug.panic(format, args) else log.warn(format, args); | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| fn unimplementedRead(comptime format: []const u8, args: anytype) u8 { | ||||
|     log.warn(format, args); | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| fn undWrite(comptime format: []const u8, args: anytype) void { | ||||
|     if (panic_on_und_io) std.debug.panic(format, args) else log.warn(format, args); | ||||
| } | ||||
|  | ||||
| /// Read / Write | ||||
| pub const PostFlag = enum(u1) { | ||||
|     FirstBoot = 0, | ||||
|     FurtherBoots = 1, | ||||
| }; | ||||
|  | ||||
| /// Write Only | ||||
| pub const HaltControl = enum { | ||||
|     Halt, | ||||
|     Stop, | ||||
|     Execute, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const DisplayControl = extern union { | ||||
|     bg_mode: Bitfield(u16, 0, 3), | ||||
|     frame_select: Bit(u16, 4), | ||||
|     hblank_interval_free: Bit(u16, 5), | ||||
|     obj_mapping: Bit(u16, 6), | ||||
|     forced_blank: Bit(u16, 7), | ||||
|     bg_enable: Bitfield(u16, 8, 4), | ||||
|     obj_enable: Bit(u16, 12), | ||||
|     win_enable: Bitfield(u16, 13, 2), | ||||
|     obj_win_enable: Bit(u16, 15), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const DisplayStatus = extern union { | ||||
|     vblank: Bit(u16, 0), | ||||
|     hblank: Bit(u16, 1), | ||||
|     coincidence: Bit(u16, 2), | ||||
|     vblank_irq: Bit(u16, 3), | ||||
|     hblank_irq: Bit(u16, 4), | ||||
|     vcount_irq: Bit(u16, 5), | ||||
|     vcount_trigger: Bitfield(u16, 8, 8), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read Only | ||||
| pub const VCount = extern union { | ||||
|     scanline: Bitfield(u16, 0, 8), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| const InterruptEnable = extern union { | ||||
|     vblank: Bit(u16, 0), | ||||
|     hblank: Bit(u16, 1), | ||||
|     coincidence: Bit(u16, 2), | ||||
|     tm0_overflow: Bit(u16, 3), | ||||
|     tm1_overflow: Bit(u16, 4), | ||||
|     tm2_overflow: Bit(u16, 5), | ||||
|     tm3_overflow: Bit(u16, 6), | ||||
|     serial: Bit(u16, 7), | ||||
|     dma0: Bit(u16, 8), | ||||
|     dma1: Bit(u16, 9), | ||||
|     dma2: Bit(u16, 10), | ||||
|     dma3: Bit(u16, 11), | ||||
|     keypad: Bit(u16, 12), | ||||
|     game_pak: Bit(u16, 13), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read Only | ||||
| /// 0 = Pressed, 1 = Released | ||||
| const KeyInput = extern union { | ||||
|     a: Bit(u16, 0), | ||||
|     b: Bit(u16, 1), | ||||
|     select: Bit(u16, 2), | ||||
|     start: Bit(u16, 3), | ||||
|     right: Bit(u16, 4), | ||||
|     left: Bit(u16, 5), | ||||
|     up: Bit(u16, 6), | ||||
|     down: Bit(u16, 7), | ||||
|     shoulder_r: Bit(u16, 8), | ||||
|     shoulder_l: Bit(u16, 9), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| // Read / Write | ||||
| pub const BackgroundControl = extern union { | ||||
|     priority: Bitfield(u16, 0, 2), | ||||
|     char_base: Bitfield(u16, 2, 2), | ||||
|     mosaic_enable: Bit(u16, 6), | ||||
|     colour_mode: Bit(u16, 7), | ||||
|     screen_base: Bitfield(u16, 8, 5), | ||||
|     display_overflow: Bit(u16, 13), | ||||
|     size: Bitfield(u16, 14, 2), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Write Only | ||||
| pub const BackgroundOffset = extern union { | ||||
|     offset: Bitfield(u16, 0, 9), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| const InterruptRequest = extern union { | ||||
|     vblank: Bit(u16, 0), | ||||
|     hblank: Bit(u16, 1), | ||||
|     coincidence: Bit(u16, 2), | ||||
|     tim0_overflow: Bit(u16, 3), | ||||
|     tim1_overflow: Bit(u16, 4), | ||||
|     tim2_overflow: Bit(u16, 5), | ||||
|     tim3_overflow: Bit(u16, 6), | ||||
|     serial: Bit(u16, 7), | ||||
|     dma0: Bit(u16, 8), | ||||
|     dma1: Bit(u16, 9), | ||||
|     dma2: Bit(u16, 10), | ||||
|     dma3: Bit(u16, 11), | ||||
|     keypad: Bit(u16, 12), | ||||
|     game_pak: Bit(u16, 13), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const DmaControl = extern union { | ||||
|     dad_adj: Bitfield(u16, 5, 2), | ||||
|     sad_adj: Bitfield(u16, 7, 2), | ||||
|     repeat: Bit(u16, 9), | ||||
|     transfer_type: Bit(u16, 10), | ||||
|     pak_drq: Bit(u16, 11), | ||||
|     start_timing: Bitfield(u16, 12, 2), | ||||
|     irq: Bit(u16, 14), | ||||
|     enabled: Bit(u16, 15), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const TimerControl = extern union { | ||||
|     frequency: Bitfield(u16, 0, 2), | ||||
|     cascade: Bit(u16, 2), | ||||
|     irq: Bit(u16, 6), | ||||
|     enabled: Bit(u16, 7), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR10 | ||||
| pub const Sweep = extern union { | ||||
|     shift: Bitfield(u8, 0, 3), | ||||
|     direction: Bit(u8, 3), | ||||
|     period: Bitfield(u8, 4, 3), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// This represents the Duty / Len | ||||
| /// NRx1 | ||||
| pub const Duty = extern union { | ||||
|     /// Write-only | ||||
|     /// Only used when bit 6 is set | ||||
|     length: Bitfield(u16, 0, 6), | ||||
|     pattern: Bitfield(u16, 6, 2), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NRx2 | ||||
| pub const Envelope = extern union { | ||||
|     period: Bitfield(u8, 0, 3), | ||||
|     direction: Bit(u8, 3), | ||||
|     init_vol: Bitfield(u8, 4, 4), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NRx3, NRx4 | ||||
| pub const Frequency = extern union { | ||||
|     /// Write-only | ||||
|     frequency: Bitfield(u16, 0, 11), | ||||
|     length_enable: Bit(u16, 14), | ||||
|     /// Write-only | ||||
|     trigger: Bit(u16, 15), | ||||
|  | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR30 | ||||
| pub const WaveSelect = extern union { | ||||
|     dimension: Bit(u8, 5), | ||||
|     bank: Bit(u8, 6), | ||||
|     enabled: Bit(u8, 7), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR32 | ||||
| pub const WaveVolume = extern union { | ||||
|     kind: Bitfield(u8, 5, 2), | ||||
|     force: Bit(u8, 7), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR43 | ||||
| pub const PolyCounter = extern union { | ||||
|     div_ratio: Bitfield(u8, 0, 3), | ||||
|     width: Bit(u8, 3), | ||||
|     shift: Bitfield(u8, 4, 4), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR44 | ||||
| pub const NoiseControl = extern union { | ||||
|     length_enable: Bit(u8, 6), | ||||
|     trigger: Bit(u8, 7), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const ChannelVolumeControl = extern union { | ||||
|     left_vol: Bitfield(u16, 0, 3), | ||||
|     right_vol: Bitfield(u16, 4, 3), | ||||
|  | ||||
|     ch1_right: Bit(u16, 8), | ||||
|     ch2_right: Bit(u16, 9), | ||||
|     ch3_right: Bit(u16, 10), | ||||
|     ch4_right: Bit(u16, 11), | ||||
|     ch1_left: Bit(u16, 12), | ||||
|     ch2_left: Bit(u16, 13), | ||||
|     ch3_left: Bit(u16, 14), | ||||
|     ch4_left: Bit(u16, 15), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const DmaSoundControl = extern union { | ||||
|     ch_vol: Bitfield(u16, 0, 2), | ||||
|     sa_vol: Bit(u16, 2), | ||||
|     sb_vol: Bit(u16, 3), | ||||
|  | ||||
|     sa_right_enable: Bit(u16, 8), | ||||
|     sa_left_enable: Bit(u16, 9), | ||||
|     sa_timer: Bit(u16, 10), | ||||
|     /// Write only? | ||||
|     sa_reset: Bit(u16, 11), | ||||
|  | ||||
|     sb_right_enable: Bit(u16, 12), | ||||
|     sb_left_enable: Bit(u16, 13), | ||||
|     sb_timer: Bit(u16, 14), | ||||
|     /// Write only? | ||||
|     sb_reset: Bit(u16, 15), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const SoundControl = extern union { | ||||
|     /// Read-only | ||||
|     ch1_enable: Bit(u8, 0), | ||||
|     /// Read-only | ||||
|     ch2_enable: Bit(u8, 1), | ||||
|     /// Read-only | ||||
|     ch3_enable: Bit(u8, 2), | ||||
|     /// Read-only | ||||
|     ch4_enable: Bit(u8, 3), | ||||
|     apu_enable: Bit(u8, 7), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const SoundBias = extern union { | ||||
|     level: Bitfield(u16, 1, 9), | ||||
|     sampling_cycle: Bitfield(u16, 14, 2), | ||||
|     raw: u16, | ||||
| }; | ||||
							
								
								
									
										157
									
								
								src/bus/timer.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								src/bus/timer.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,157 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const TimerControl = @import("io.zig").TimerControl; | ||||
| const Io = @import("io.zig").Io; | ||||
| const Scheduler = @import("../scheduler.zig").Scheduler; | ||||
| const Event = @import("../scheduler.zig").Event; | ||||
| const Arm7tdmi = @import("../cpu.zig").Arm7tdmi; | ||||
|  | ||||
| const log = std.log.scoped(.Timer); | ||||
|  | ||||
| pub const Timers = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     _0: Timer(0), | ||||
|     _1: Timer(1), | ||||
|     _2: Timer(2), | ||||
|     _3: Timer(3), | ||||
|  | ||||
|     pub fn init(sched: *Scheduler) Self { | ||||
|         return .{ | ||||
|             ._0 = Timer(0).init(sched), | ||||
|             ._1 = Timer(1).init(sched), | ||||
|             ._2 = Timer(2).init(sched), | ||||
|             ._3 = Timer(3).init(sched), | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| fn Timer(comptime id: u2) type { | ||||
|     return struct { | ||||
|         const Self = @This(); | ||||
|  | ||||
|         /// Read Only, Internal. Please use self.counter() | ||||
|         _counter: u16, | ||||
|  | ||||
|         /// Write Only, Internal. Please use self.writeCntLow() | ||||
|         _reload: u16, | ||||
|  | ||||
|         /// Write Only, Internal. Please use self.WriteCntHigh() | ||||
|         cnt: TimerControl, | ||||
|  | ||||
|         /// Internal. | ||||
|         sched: *Scheduler, | ||||
|  | ||||
|         /// Internal | ||||
|         _start_timestamp: u64, | ||||
|  | ||||
|         pub fn init(sched: *Scheduler) Self { | ||||
|             return .{ | ||||
|                 ._reload = 0, | ||||
|                 ._counter = 0, | ||||
|                 .cnt = .{ .raw = 0x0000 }, | ||||
|                 .sched = sched, | ||||
|                 ._start_timestamp = 0, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         pub fn counter(self: *const Self) u16 { | ||||
|             if (self.cnt.cascade.read()) | ||||
|                 return self._counter | ||||
|             else | ||||
|                 return self._counter +% @truncate(u16, (self.sched.now() - self._start_timestamp) / self.frequency()); | ||||
|         } | ||||
|  | ||||
|         pub fn writeCnt(self: *Self, word: u32) void { | ||||
|             self.writeCntLow(@truncate(u16, word)); | ||||
|             self.writeCntHigh(@truncate(u16, word >> 16)); | ||||
|         } | ||||
|  | ||||
|         pub fn writeCntLow(self: *Self, halfword: u16) void { | ||||
|             self._reload = halfword; | ||||
|         } | ||||
|  | ||||
|         pub fn writeCntHigh(self: *Self, halfword: u16) void { | ||||
|             const new = TimerControl{ .raw = halfword }; | ||||
|  | ||||
|             // If Timer happens to be enabled, It will either be resheduled or disabled | ||||
|             self.sched.removeScheduledEvent(.{ .TimerOverflow = id }); | ||||
|  | ||||
|             if (!self.cnt.enabled.read() and new.enabled.read()) { | ||||
|                 // Reload on Rising edge | ||||
|                 self._counter = self._reload; | ||||
|  | ||||
|                 if (!new.cascade.read()) self.scheduleOverflow(0); | ||||
|             } | ||||
|  | ||||
|             self.cnt.raw = halfword; | ||||
|         } | ||||
|  | ||||
|         pub fn handleOverflow(self: *Self, cpu: *Arm7tdmi, late: u64) void { | ||||
|             // Fire IRQ if enabled | ||||
|             const io = &cpu.bus.io; | ||||
|             const tim = &cpu.bus.tim; | ||||
|  | ||||
|             if (self.cnt.irq.read()) { | ||||
|                 switch (id) { | ||||
|                     0 => io.irq.tim0_overflow.set(), | ||||
|                     1 => io.irq.tim1_overflow.set(), | ||||
|                     2 => io.irq.tim2_overflow.set(), | ||||
|                     3 => io.irq.tim3_overflow.set(), | ||||
|                 } | ||||
|  | ||||
|                 cpu.handleInterrupt(); | ||||
|             } | ||||
|  | ||||
|             // DMA Sound Things | ||||
|             if (id == 0 or id == 1) { | ||||
|                 const apu = &cpu.bus.apu; | ||||
|  | ||||
|                 const a_tim = @boolToInt(apu.dma_cnt.sa_timer.read()); | ||||
|                 const b_tim = @boolToInt(apu.dma_cnt.sb_timer.read()); | ||||
|  | ||||
|                 if (a_tim == id) apu.handleTimerOverflow(.A, cpu); | ||||
|                 if (b_tim == id) apu.handleTimerOverflow(.B, cpu); | ||||
|             } | ||||
|  | ||||
|             // Perform Cascade Behaviour | ||||
|             switch (id) { | ||||
|                 0 => if (tim._1.cnt.cascade.read()) { | ||||
|                     tim._1._counter +%= 1; | ||||
|                     if (tim._1._counter == 0) tim._1.handleOverflow(cpu, late); | ||||
|                 }, | ||||
|                 1 => if (tim._2.cnt.cascade.read()) { | ||||
|                     tim._2._counter +%= 1; | ||||
|                     if (tim._2._counter == 0) tim._2.handleOverflow(cpu, late); | ||||
|                 }, | ||||
|                 2 => if (tim._3.cnt.cascade.read()) { | ||||
|                     tim._3._counter +%= 1; | ||||
|                     if (tim._3._counter == 0) tim._3.handleOverflow(cpu, late); | ||||
|                 }, | ||||
|                 3 => {}, // There is no Timer for TIM3 to "cascade" to, | ||||
|             } | ||||
|  | ||||
|             // Reschedule Timer if we're not cascading | ||||
|             if (!self.cnt.cascade.read()) { | ||||
|                 self._counter = self._reload; | ||||
|                 self.scheduleOverflow(late); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fn scheduleOverflow(self: *Self, late: u64) void { | ||||
|             const when = (@as(u64, 0x10000) - self._counter) * self.frequency(); | ||||
|  | ||||
|             self._start_timestamp = self.sched.now(); | ||||
|             self.sched.push(.{ .TimerOverflow = id }, self.sched.now() + when - late); | ||||
|         } | ||||
|  | ||||
|         fn frequency(self: *const Self) u16 { | ||||
|             return switch (self.cnt.frequency.read()) { | ||||
|                 0 => 1, | ||||
|                 1 => 64, | ||||
|                 2 => 256, | ||||
|                 3 => 1024, | ||||
|             }; | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| @@ -1,86 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const toml = @import("toml"); | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
|  | ||||
| const log = std.log.scoped(.Config); | ||||
| var state: Config = .{}; | ||||
|  | ||||
| const Config = struct { | ||||
|     host: Host = .{}, | ||||
|     guest: Guest = .{}, | ||||
|     debug: Debug = .{}, | ||||
|  | ||||
|     /// Settings related to the Computer the Emulator is being run on | ||||
|     const Host = struct { | ||||
|         /// Using Nearest-Neighbor, multiply the resolution of the GBA Window | ||||
|         win_scale: i64 = 3, | ||||
|         /// Enable Vsync | ||||
|         /// | ||||
|         /// Note: This does not affect whether Emulation is synced to 59Hz | ||||
|         vsync: bool = true, | ||||
|         /// Mute ZBA | ||||
|         mute: bool = false, | ||||
|     }; | ||||
|  | ||||
|     // Settings realted to the emulation itself | ||||
|     const Guest = struct { | ||||
|         /// Whether Emulation thread to sync to Audio Callbacks | ||||
|         audio_sync: bool = true, | ||||
|         /// Whether Emulation thread should sync to 59Hz | ||||
|         video_sync: bool = true, | ||||
|         /// Whether RTC I/O should always be enabled | ||||
|         force_rtc: bool = false, | ||||
|         /// Skip BIOS | ||||
|         skip_bios: bool = false, | ||||
|     }; | ||||
|  | ||||
|     /// Settings related to debugging ZBA | ||||
|     const Debug = struct { | ||||
|         /// Enable CPU Trace logs | ||||
|         cpu_trace: bool = false, | ||||
|         /// If false and ZBA is built in debug mode, ZBA will panic on unhandled I/O | ||||
|         unhandled_io: bool = true, | ||||
|     }; | ||||
| }; | ||||
|  | ||||
| pub fn config() *const Config { | ||||
|     return &state; | ||||
| } | ||||
|  | ||||
| /// Reads a config file and then loads it into the global state | ||||
| pub fn load(allocator: Allocator, file_path: []const u8) !void { | ||||
|     var config_file = try std.fs.cwd().openFile(file_path, .{}); | ||||
|     defer config_file.close(); | ||||
|  | ||||
|     log.info("loaded from {s}", .{file_path}); | ||||
|  | ||||
|     const contents = try config_file.readToEndAlloc(allocator, try config_file.getEndPos()); | ||||
|     defer allocator.free(contents); | ||||
|  | ||||
|     var parser = try toml.parseFile(allocator, file_path); | ||||
|     defer parser.deinit(); | ||||
|  | ||||
|     const table = try parser.parse(); | ||||
|     defer table.deinit(); | ||||
|  | ||||
|     // TODO: Report unknown config options | ||||
|  | ||||
|     if (table.keys.get("Host")) |host| { | ||||
|         if (host.Table.keys.get("win_scale")) |scale| state.host.win_scale = scale.Integer; | ||||
|         if (host.Table.keys.get("vsync")) |vsync| state.host.vsync = vsync.Boolean; | ||||
|         if (host.Table.keys.get("mute")) |mute| state.host.mute = mute.Boolean; | ||||
|     } | ||||
|  | ||||
|     if (table.keys.get("Guest")) |guest| { | ||||
|         if (guest.Table.keys.get("audio_sync")) |sync| state.guest.audio_sync = sync.Boolean; | ||||
|         if (guest.Table.keys.get("video_sync")) |sync| state.guest.video_sync = sync.Boolean; | ||||
|         if (guest.Table.keys.get("force_rtc")) |forced| state.guest.force_rtc = forced.Boolean; | ||||
|         if (guest.Table.keys.get("skip_bios")) |skip| state.guest.skip_bios = skip.Boolean; | ||||
|     } | ||||
|  | ||||
|     if (table.keys.get("Debug")) |debug| { | ||||
|         if (debug.Table.keys.get("cpu_trace")) |trace| state.debug.cpu_trace = trace.Boolean; | ||||
|         if (debug.Table.keys.get("unhandled_io")) |unhandled| state.debug.unhandled_io = unhandled.Boolean; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										439
									
								
								src/core/Bus.zig
									
									
									
									
									
								
							
							
						
						
									
										439
									
								
								src/core/Bus.zig
									
									
									
									
									
								
							| @@ -1,439 +0,0 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Arm7tdmi = @import("cpu.zig").Arm7tdmi; | ||||
| const Bios = @import("bus/Bios.zig"); | ||||
| const Ewram = @import("bus/Ewram.zig"); | ||||
| const GamePak = @import("bus/GamePak.zig"); | ||||
| const Io = @import("bus/io.zig").Io; | ||||
| const Iwram = @import("bus/Iwram.zig"); | ||||
| const Ppu = @import("ppu.zig").Ppu; | ||||
| const Apu = @import("apu.zig").Apu; | ||||
| const DmaTuple = @import("bus/dma.zig").DmaTuple; | ||||
| const TimerTuple = @import("bus/timer.zig").TimerTuple; | ||||
| const Scheduler = @import("scheduler.zig").Scheduler; | ||||
| const FilePaths = @import("../util.zig").FilePaths; | ||||
|  | ||||
| const io = @import("bus/io.zig"); | ||||
| const Allocator = std.mem.Allocator; | ||||
| const log = std.log.scoped(.Bus); | ||||
|  | ||||
| const createDmaTuple = @import("bus/dma.zig").create; | ||||
| const createTimerTuple = @import("bus/timer.zig").create; | ||||
| const rotr = @import("../util.zig").rotr; | ||||
|  | ||||
| const timings: [2][0x10]u8 = [_][0x10]u8{ | ||||
|     // BIOS, Unused, EWRAM, IWRAM, I/0, PALRAM, VRAM, OAM, ROM0, ROM0, ROM1, ROM1, ROM2, ROM2, SRAM, Unused | ||||
|     [_]u8{ 1, 1, 3, 1, 1, 1, 1, 1, 5, 5, 5, 5, 5, 5, 5, 5 }, // 8-bit & 16-bit | ||||
|     [_]u8{ 1, 1, 6, 1, 1, 2, 2, 1, 8, 8, 8, 8, 8, 8, 8, 8 }, // 32-bit | ||||
| }; | ||||
|  | ||||
| pub const fetch_timings: [2][0x10]u8 = [_][0x10]u8{ | ||||
|     // BIOS, Unused, EWRAM, IWRAM, I/0, PALRAM, VRAM, OAM, ROM0, ROM0, ROM1, ROM1, ROM2, ROM2, SRAM, Unused | ||||
|     [_]u8{ 1, 1, 3, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 5, 5 }, // 8-bit & 16-bit | ||||
|     [_]u8{ 1, 1, 6, 1, 1, 2, 2, 1, 4, 4, 4, 4, 4, 4, 8, 8 }, // 32-bit | ||||
| }; | ||||
|  | ||||
| // Fastmem Related | ||||
| const page_size = 1 * 0x400; // 1KiB | ||||
| const address_space_size = 0x1000_0000; | ||||
| const table_len = address_space_size / page_size; | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| pak: GamePak, | ||||
| bios: Bios, | ||||
| ppu: Ppu, | ||||
| apu: Apu, | ||||
| dma: DmaTuple, | ||||
| tim: TimerTuple, | ||||
| iwram: Iwram, | ||||
| ewram: Ewram, | ||||
| io: Io, | ||||
|  | ||||
| cpu: *Arm7tdmi, | ||||
| sched: *Scheduler, | ||||
|  | ||||
| read_table: *const [table_len]?*const anyopaque, | ||||
| write_tables: [2]*const [table_len]?*anyopaque, | ||||
| allocator: Allocator, | ||||
|  | ||||
| pub fn init(self: *Self, allocator: Allocator, sched: *Scheduler, cpu: *Arm7tdmi, paths: FilePaths) !void { | ||||
|     const tables = try allocator.alloc(?*anyopaque, 3 * table_len); // Allocate all tables | ||||
|  | ||||
|     const read_table: *[table_len]?*const anyopaque = tables[0..table_len]; | ||||
|     const left_write: *[table_len]?*anyopaque = tables[table_len .. 2 * table_len]; | ||||
|     const right_write: *[table_len]?*anyopaque = tables[2 * table_len .. 3 * table_len]; | ||||
|  | ||||
|     self.* = .{ | ||||
|         .pak = try GamePak.init(allocator, cpu, paths.rom, paths.save), | ||||
|         .bios = try Bios.init(allocator, paths.bios), | ||||
|         .ppu = try Ppu.init(allocator, sched), | ||||
|         .apu = Apu.init(sched), | ||||
|         .iwram = try Iwram.init(allocator), | ||||
|         .ewram = try Ewram.init(allocator), | ||||
|         .dma = createDmaTuple(), | ||||
|         .tim = createTimerTuple(sched), | ||||
|         .io = Io.init(), | ||||
|         .cpu = cpu, | ||||
|         .sched = sched, | ||||
|  | ||||
|         .read_table = read_table, | ||||
|         .write_tables = .{ left_write, right_write }, | ||||
|         .allocator = allocator, | ||||
|     }; | ||||
|  | ||||
|     // read_table, write_tables, and *Self are not restricted to the lifetime | ||||
|     // of this init function so we can initialize our tables here | ||||
|     fillReadTable(self, read_table); | ||||
|  | ||||
|     // Internal Display Memory behavious unusually on 8-bit reads | ||||
|     // so we have two different tables depending on whether there's an 8-bit read or not | ||||
|     fillWriteTable(u32, self, left_write); | ||||
|     fillWriteTable(u8, self, right_write); | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: *Self) void { | ||||
|     self.iwram.deinit(); | ||||
|     self.ewram.deinit(); | ||||
|     self.pak.deinit(); | ||||
|     self.bios.deinit(); | ||||
|     self.ppu.deinit(); | ||||
|  | ||||
|     // This is so I can deallocate the original `allocator.alloc`. I have to re-make the type | ||||
|     // since I'm not keeping it around, This is very jank and bad though | ||||
|     // FIXME: please figure out another way | ||||
|     self.allocator.free(@ptrCast([*]const ?*anyopaque, self.read_table[0..])[0 .. 3 * table_len]); | ||||
|     self.* = undefined; | ||||
| } | ||||
|  | ||||
| fn fillReadTable(bus: *Self, table: *[table_len]?*const anyopaque) void { | ||||
|     const vramMirror = @import("ppu.zig").Vram.mirror; | ||||
|  | ||||
|     for (table) |*ptr, i| { | ||||
|         const addr = page_size * i; | ||||
|  | ||||
|         ptr.* = switch (addr) { | ||||
|             // General Internal Memory | ||||
|             0x0000_0000...0x0000_3FFF => null, // BIOS has it's own checks | ||||
|             0x0200_0000...0x02FF_FFFF => &bus.ewram.buf[addr & 0x3FFFF], | ||||
|             0x0300_0000...0x03FF_FFFF => &bus.iwram.buf[addr & 0x7FFF], | ||||
|             0x0400_0000...0x0400_03FF => null, // I/O | ||||
|  | ||||
|             // Internal Display Memory | ||||
|             0x0500_0000...0x05FF_FFFF => &bus.ppu.palette.buf[addr & 0x3FF], | ||||
|             0x0600_0000...0x06FF_FFFF => &bus.ppu.vram.buf[vramMirror(addr)], | ||||
|             0x0700_0000...0x07FF_FFFF => &bus.ppu.oam.buf[addr & 0x3FF], | ||||
|  | ||||
|             // External Memory (Game Pak) | ||||
|             0x0800_0000...0x0DFF_FFFF => fillTableExternalMemory(bus, addr), | ||||
|             0x0E00_0000...0x0FFF_FFFF => null, // SRAM | ||||
|             else => null, | ||||
|         }; | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn fillWriteTable(comptime T: type, bus: *Self, table: *[table_len]?*const anyopaque) void { | ||||
|     comptime std.debug.assert(T == u32 or T == u16 or T == u8); | ||||
|     const vramMirror = @import("ppu.zig").Vram.mirror; | ||||
|  | ||||
|     for (table) |*ptr, i| { | ||||
|         const addr = page_size * i; | ||||
|  | ||||
|         ptr.* = switch (addr) { | ||||
|             // General Internal Memory | ||||
|             0x0000_0000...0x0000_3FFF => null, // BIOS has it's own checks | ||||
|             0x0200_0000...0x02FF_FFFF => &bus.ewram.buf[addr & 0x3FFFF], | ||||
|             0x0300_0000...0x03FF_FFFF => &bus.iwram.buf[addr & 0x7FFF], | ||||
|             0x0400_0000...0x0400_03FF => null, // I/O | ||||
|  | ||||
|             // Internal Display Memory | ||||
|             0x0500_0000...0x05FF_FFFF => if (T != u8) &bus.ppu.palette.buf[addr & 0x3FF] else null, | ||||
|             0x0600_0000...0x06FF_FFFF => if (T != u8) &bus.ppu.vram.buf[vramMirror(addr)] else null, | ||||
|             0x0700_0000...0x07FF_FFFF => if (T != u8) &bus.ppu.oam.buf[addr & 0x3FF] else null, | ||||
|  | ||||
|             // External Memory (Game Pak) | ||||
|             0x0800_0000...0x0DFF_FFFF => null, // ROM | ||||
|             0x0E00_0000...0x0FFF_FFFF => null, // SRAM | ||||
|             else => null, | ||||
|         }; | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn fillTableExternalMemory(bus: *Self, addr: usize) ?*anyopaque { | ||||
|     // see `GamePak.zig` for more information about what conditions need to be true | ||||
|     // so that a simple pointer dereference isn't possible | ||||
|  | ||||
|     const start_addr = addr; | ||||
|     const end_addr = addr + page_size; | ||||
|  | ||||
|     const gpio_data = start_addr <= 0x0800_00C4 and 0x0800_00C4 < end_addr; | ||||
|     const gpio_direction = start_addr <= 0x0800_00C6 and 0x0800_00C6 < end_addr; | ||||
|     const gpio_control = start_addr <= 0x0800_00C8 and 0x0800_00C8 < end_addr; | ||||
|  | ||||
|     if (bus.pak.gpio.device.kind != .None and (gpio_data or gpio_direction or gpio_control)) { | ||||
|         // We found a GPIO device, and this page a GPIO register. We want to handle this in slowmem | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     if (bus.pak.backup.kind == .Eeprom) { | ||||
|         if (bus.pak.buf.len > 0x100_000) { | ||||
|             // We are using a "large" EEPROM which means that if the below check is true | ||||
|             // this page has an address that's reserved for the EEPROM and therefore must | ||||
|             // be handled in slowmem | ||||
|             if (addr & 0x1FF_FFFF > 0x1FF_FEFF) return null; | ||||
|         } else { | ||||
|             // We are using a "small" EEPROM which means that if the below check is true | ||||
|             // (that is, we're in the 0xD address page) then we must handle at least one | ||||
|             // address in this page in slowmem | ||||
|             if (@truncate(u4, addr >> 24) == 0xD) return null; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Finally, the GamePak has some unique behaviour for reads past the end of the ROM, | ||||
|     // so those will be handled by slowmem as well | ||||
|     const masked_addr = addr & 0x1FF_FFFF; | ||||
|     if (masked_addr >= bus.pak.buf.len) return null; | ||||
|  | ||||
|     return &bus.pak.buf[masked_addr]; | ||||
| } | ||||
|  | ||||
| // TODO: Take advantage of fastmem here too? | ||||
| pub fn dbgRead(self: *const Self, comptime T: type, unaligned_address: u32) T { | ||||
|     const page = @truncate(u8, unaligned_address >> 24); | ||||
|     const address = forceAlign(T, unaligned_address); | ||||
|  | ||||
|     return switch (page) { | ||||
|         // General Internal Memory | ||||
|         0x00 => blk: { | ||||
|             if (address < Bios.size) | ||||
|                 break :blk self.bios.dbgRead(T, self.cpu.r[15], address); | ||||
|  | ||||
|             break :blk self.openBus(T, address); | ||||
|         }, | ||||
|         0x02 => self.ewram.read(T, address), | ||||
|         0x03 => self.iwram.read(T, address), | ||||
|         0x04 => self.readIo(T, address), | ||||
|  | ||||
|         // Internal Display Memory | ||||
|         0x05 => self.ppu.palette.read(T, address), | ||||
|         0x06 => self.ppu.vram.read(T, address), | ||||
|         0x07 => self.ppu.oam.read(T, address), | ||||
|  | ||||
|         // External Memory (Game Pak) | ||||
|         0x08...0x0D => self.pak.dbgRead(T, address), | ||||
|         0x0E...0x0F => blk: { | ||||
|             const value = self.pak.backup.read(unaligned_address); | ||||
|  | ||||
|             const multiplier = switch (T) { | ||||
|                 u32 => 0x01010101, | ||||
|                 u16 => 0x0101, | ||||
|                 u8 => 1, | ||||
|                 else => @compileError("Backup: Unsupported read width"), | ||||
|             }; | ||||
|  | ||||
|             break :blk @as(T, value) * multiplier; | ||||
|         }, | ||||
|         else => self.openBus(T, address), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| fn readIo(self: *const Self, comptime T: type, address: u32) T { | ||||
|     return io.read(self, T, address) orelse self.openBus(T, address); | ||||
| } | ||||
|  | ||||
| fn openBus(self: *const Self, comptime T: type, address: u32) T { | ||||
|     @setCold(true); | ||||
|     const r15 = self.cpu.r[15]; | ||||
|  | ||||
|     const word = blk: { | ||||
|         // If Arm, get the most recently fetched instruction (PC + 8) | ||||
|         // | ||||
|         // FIXME: This is most likely a faulty assumption. | ||||
|         // I think what *actually* happens is that the Bus has a latch for the most | ||||
|         // recently fetched piece of data, which is then returned during Open Bus (also DMA open bus?) | ||||
|         // I can "get away" with this because it's very statistically likely that the most recently latched value is | ||||
|         // the most recently fetched instruction by the pipeline | ||||
|         if (!self.cpu.cpsr.t.read()) break :blk self.cpu.pipe.stage[1].?; | ||||
|  | ||||
|         const page = @truncate(u8, r15 >> 24); | ||||
|  | ||||
|         // PC + 2 = stage[0] | ||||
|         // PC + 4 = stage[1] | ||||
|         // PC + 6 = Need a Debug Read for this? | ||||
|  | ||||
|         switch (page) { | ||||
|             // EWRAM, PALRAM, VRAM, and Game ROM (16-bit) | ||||
|             0x02, 0x05, 0x06, 0x08...0x0D => { | ||||
|                 const halfword: u32 = @truncate(u16, self.cpu.pipe.stage[1].?); | ||||
|                 break :blk halfword << 16 | halfword; | ||||
|             }, | ||||
|  | ||||
|             // BIOS or OAM (32-bit) | ||||
|             0x00, 0x07 => { | ||||
|                 // Aligned: (PC + 6) | (PC + 4) | ||||
|                 // Unaligned: (PC + 4) | (PC + 2) | ||||
|                 const aligned = address & 3 == 0b00; | ||||
|  | ||||
|                 // TODO: What to do on PC + 6? | ||||
|                 const high: u32 = if (aligned) self.dbgRead(u16, r15 + 4) else @truncate(u16, self.cpu.pipe.stage[1].?); | ||||
|                 const low: u32 = @truncate(u16, self.cpu.pipe.stage[@boolToInt(aligned)].?); | ||||
|  | ||||
|                 break :blk high << 16 | low; | ||||
|             }, | ||||
|  | ||||
|             // IWRAM (16-bit but special) | ||||
|             0x03 => { | ||||
|                 // Aligned: (PC + 2) | (PC + 4) | ||||
|                 // Unaligned: (PC + 4) | (PC + 2) | ||||
|                 const aligned = address & 3 == 0b00; | ||||
|  | ||||
|                 const high: u32 = @truncate(u16, self.cpu.pipe.stage[1 - @boolToInt(aligned)].?); | ||||
|                 const low: u32 = @truncate(u16, self.cpu.pipe.stage[@boolToInt(aligned)].?); | ||||
|  | ||||
|                 break :blk high << 16 | low; | ||||
|             }, | ||||
|             else => { | ||||
|                 log.err("THUMB open bus read from 0x{X:0>2} page @0x{X:0>8}", .{ page, address }); | ||||
|                 @panic("invariant most-likely broken"); | ||||
|             }, | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     return @truncate(T, word); | ||||
| } | ||||
|  | ||||
| pub fn read(self: *Self, comptime T: type, unaligned_address: u32) T { | ||||
|     const bits = @typeInfo(std.math.IntFittingRange(0, page_size - 1)).Int.bits; | ||||
|     const page = unaligned_address >> bits; | ||||
|     const offset = unaligned_address & (page_size - 1); | ||||
|  | ||||
|     // whether or not we do this in slowmem or fastmem, we should advance the scheduler | ||||
|     self.sched.tick += timings[@boolToInt(T == u32)][@truncate(u4, unaligned_address >> 24)]; | ||||
|  | ||||
|     // We're doing some serious out-of-bounds open-bus reads | ||||
|     if (page >= table_len) return self.openBus(T, unaligned_address); | ||||
|  | ||||
|     if (self.read_table[page]) |some_ptr| { | ||||
|         // We have a pointer to a page, cast the pointer to it's underlying type | ||||
|         const Ptr = [*]const T; | ||||
|         const alignment = @alignOf(std.meta.Child(Ptr)); | ||||
|         const ptr = @ptrCast(Ptr, @alignCast(alignment, some_ptr)); | ||||
|  | ||||
|         // Note: We don't check array length, since we force align the | ||||
|         // lower bits of the address as the GBA would | ||||
|         return ptr[forceAlign(T, offset) / @sizeOf(T)]; | ||||
|     } | ||||
|  | ||||
|     return self.slowRead(T, unaligned_address); | ||||
| } | ||||
|  | ||||
| fn slowRead(self: *Self, comptime T: type, unaligned_address: u32) T { | ||||
|     @setCold(true); | ||||
|  | ||||
|     const page = @truncate(u8, unaligned_address >> 24); | ||||
|     const address = forceAlign(T, unaligned_address); | ||||
|  | ||||
|     return switch (page) { | ||||
|         // General Internal Memory | ||||
|         0x00 => blk: { | ||||
|             if (address < Bios.size) | ||||
|                 break :blk self.bios.read(T, self.cpu.r[15], address); | ||||
|  | ||||
|             break :blk self.openBus(T, address); | ||||
|         }, | ||||
|         0x02 => unreachable, // completely handled by fastmeme | ||||
|         0x03 => unreachable, // completely handled by fastmeme | ||||
|         0x04 => self.readIo(T, address), | ||||
|  | ||||
|         // Internal Display Memory | ||||
|         0x05 => unreachable, // completely handled by fastmeme | ||||
|         0x06 => unreachable, // completely handled by fastmeme | ||||
|         0x07 => unreachable, // completely handled by fastmeme | ||||
|  | ||||
|         // External Memory (Game Pak) | ||||
|         0x08...0x0D => self.pak.read(T, address), | ||||
|         0x0E...0x0F => blk: { | ||||
|             const value = self.pak.backup.read(unaligned_address); | ||||
|  | ||||
|             const multiplier = switch (T) { | ||||
|                 u32 => 0x01010101, | ||||
|                 u16 => 0x0101, | ||||
|                 u8 => 1, | ||||
|                 else => @compileError("Backup: Unsupported read width"), | ||||
|             }; | ||||
|  | ||||
|             break :blk @as(T, value) * multiplier; | ||||
|         }, | ||||
|         else => self.openBus(T, address), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(self: *Self, comptime T: type, unaligned_address: u32, value: T) void { | ||||
|     const bits = @typeInfo(std.math.IntFittingRange(0, page_size - 1)).Int.bits; | ||||
|     const page = unaligned_address >> bits; | ||||
|     const offset = unaligned_address & (page_size - 1); | ||||
|  | ||||
|     // whether or not we do this in slowmem or fastmem, we should advance the scheduler | ||||
|     self.sched.tick += timings[@boolToInt(T == u32)][@truncate(u4, unaligned_address >> 24)]; | ||||
|  | ||||
|     // We're doing some serious out-of-bounds open-bus writes, they do nothing though | ||||
|     if (page >= table_len) return; | ||||
|  | ||||
|     if (self.write_tables[@boolToInt(T == u8)][page]) |some_ptr| { | ||||
|         // We have a pointer to a page, cast the pointer to it's underlying type | ||||
|         const Ptr = [*]T; | ||||
|         const alignment = @alignOf(std.meta.Child(Ptr)); | ||||
|         const ptr = @ptrCast(Ptr, @alignCast(alignment, some_ptr)); | ||||
|  | ||||
|         // Note: We don't check array length, since we force align the | ||||
|         // lower bits of the address as the GBA would | ||||
|         ptr[forceAlign(T, offset) / @sizeOf(T)] = value; | ||||
|     } else { | ||||
|         // we can return early if this is an 8-bit OAM write | ||||
|         if (T == u8 and @truncate(u8, unaligned_address >> 24) == 0x07) return; | ||||
|  | ||||
|         self.slowWrite(T, unaligned_address, value); | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn slowWrite(self: *Self, comptime T: type, unaligned_address: u32, value: T) void { | ||||
|     // @setCold(true); | ||||
|     const page = @truncate(u8, unaligned_address >> 24); | ||||
|     const address = forceAlign(T, unaligned_address); | ||||
|  | ||||
|     switch (page) { | ||||
|         // General Internal Memory | ||||
|         0x00 => self.bios.write(T, address, value), | ||||
|         0x02 => unreachable, // completely handled by fastmem | ||||
|         0x03 => unreachable, // completely handled by fastmem | ||||
|         0x04 => io.write(self, T, address, value), | ||||
|  | ||||
|         // Internal Display Memory | ||||
|         0x05 => self.ppu.palette.write(T, address, value), | ||||
|         0x06 => self.ppu.vram.write(T, self.ppu.dispcnt, address, value), | ||||
|         0x07 => unreachable, // completely handled by fastmem | ||||
|  | ||||
|         // External Memory (Game Pak) | ||||
|         0x08...0x0D => self.pak.write(T, self.dma[3].word_count, address, value), | ||||
|         0x0E...0x0F => self.pak.backup.write(unaligned_address, @truncate(u8, rotr(T, value, 8 * rotateBy(T, unaligned_address)))), | ||||
|         else => {}, | ||||
|     } | ||||
| } | ||||
|  | ||||
| inline fn rotateBy(comptime T: type, address: u32) u32 { | ||||
|     return switch (T) { | ||||
|         u32 => address & 3, | ||||
|         u16 => address & 1, | ||||
|         u8 => 0, | ||||
|         else => @compileError("Backup: Unsupported write width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| inline fn forceAlign(comptime T: type, address: u32) u32 { | ||||
|     return switch (T) { | ||||
|         u32 => address & ~@as(u32, 3), | ||||
|         u16 => address & ~@as(u32, 1), | ||||
|         u8 => address, | ||||
|         else => @compileError("Bus: Invalid read/write type"), | ||||
|     }; | ||||
| } | ||||
							
								
								
									
										561
									
								
								src/core/apu.zig
									
									
									
									
									
								
							
							
						
						
									
										561
									
								
								src/core/apu.zig
									
									
									
									
									
								
							| @@ -1,561 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const SDL = @import("sdl2"); | ||||
| const io = @import("bus/io.zig"); | ||||
| const util = @import("../util.zig"); | ||||
|  | ||||
| const Arm7tdmi = @import("cpu.zig").Arm7tdmi; | ||||
| const Scheduler = @import("scheduler.zig").Scheduler; | ||||
| const ToneSweep = @import("apu/ToneSweep.zig"); | ||||
| const Tone = @import("apu/Tone.zig"); | ||||
| const Wave = @import("apu/Wave.zig"); | ||||
| const Noise = @import("apu/Noise.zig"); | ||||
|  | ||||
| const SoundFifo = std.fifo.LinearFifo(u8, .{ .Static = 0x20 }); | ||||
|  | ||||
| const getHalf = util.getHalf; | ||||
| const setHalf = util.setHalf; | ||||
| const intToBytes = util.intToBytes; | ||||
| const RingBuffer = util.RingBuffer; | ||||
|  | ||||
| const log = std.log.scoped(.APU); | ||||
|  | ||||
| pub fn read(comptime T: type, apu: *const Apu, addr: u32) ?T { | ||||
|     const byte_addr = @truncate(u8, addr); | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => switch (byte_addr) { | ||||
|             0x60 => @as(T, apu.ch1.sound1CntH()) << 16 | apu.ch1.sound1CntL(), | ||||
|             0x64 => apu.ch1.sound1CntX(), | ||||
|             0x68 => apu.ch2.sound2CntL(), | ||||
|             0x6C => apu.ch2.sound2CntH(), | ||||
|             0x70 => @as(T, apu.ch3.sound3CntH()) << 16 | apu.ch3.sound3CntL(), | ||||
|             0x74 => apu.ch3.sound3CntX(), | ||||
|             0x78 => apu.ch4.sound4CntL(), | ||||
|             0x7C => apu.ch4.sound4CntH(), | ||||
|             0x80 => @as(T, apu.dma_cnt.raw) << 16 | apu.psg_cnt.raw, // SOUNDCNT_H, SOUNDCNT_L | ||||
|             0x84 => apu.soundCntX(), | ||||
|             0x88 => apu.bias.raw, // SOUNDBIAS, high is unused | ||||
|             0x8C => null, | ||||
|             0x90, 0x94, 0x98, 0x9C => apu.ch3.wave_dev.read(T, apu.ch3.select, addr), | ||||
|             0xA0 => null, // FIFO_A | ||||
|             0xA4 => null, // FIFO_B | ||||
|             else => util.io.read.err(T, log, "unaligned {} read from 0x{X:0>8}", .{ T, addr }), | ||||
|         }, | ||||
|         u16 => switch (byte_addr) { | ||||
|             0x60 => apu.ch1.sound1CntL(), | ||||
|             0x62 => apu.ch1.sound1CntH(), | ||||
|             0x64 => apu.ch1.sound1CntX(), | ||||
|             0x66 => 0x0000, // suite.gba expects 0x0000, not 0xDEAD | ||||
|             0x68 => apu.ch2.sound2CntL(), | ||||
|             0x6A => 0x0000, | ||||
|             0x6C => apu.ch2.sound2CntH(), | ||||
|             0x6E => 0x0000, | ||||
|             0x70 => apu.ch3.sound3CntL(), | ||||
|             0x72 => apu.ch3.sound3CntH(), | ||||
|             0x74 => apu.ch3.sound3CntX(), | ||||
|             0x76 => 0x0000, | ||||
|             0x78 => apu.ch4.sound4CntL(), | ||||
|             0x7A => 0x0000, | ||||
|             0x7C => apu.ch4.sound4CntH(), | ||||
|             0x7E => 0x0000, | ||||
|             0x80 => apu.soundCntL(), | ||||
|             0x82 => apu.soundCntH(), | ||||
|             0x84 => apu.soundCntX(), | ||||
|             0x86 => 0x0000, | ||||
|             0x88 => apu.bias.raw, // SOUNDBIAS | ||||
|             0x8A => 0x0000, | ||||
|             0x8C, 0x8E => null, | ||||
|             0x90, 0x92, 0x94, 0x96, 0x98, 0x9A, 0x9C, 0x9E => apu.ch3.wave_dev.read(T, apu.ch3.select, addr), | ||||
|             0xA0, 0xA2 => null, // FIFO_A | ||||
|             0xA4, 0xA6 => null, // FIFO_B | ||||
|             else => util.io.read.err(T, log, "unaligned {} read from 0x{X:0>8}", .{ T, addr }), | ||||
|         }, | ||||
|         u8 => switch (byte_addr) { | ||||
|             0x60, 0x61 => @truncate(T, @as(u16, apu.ch1.sound1CntL()) >> getHalf(byte_addr)), | ||||
|             0x62, 0x63 => @truncate(T, apu.ch1.sound1CntH() >> getHalf(byte_addr)), | ||||
|             0x64, 0x65 => @truncate(T, apu.ch1.sound1CntX() >> getHalf(byte_addr)), | ||||
|             0x66, 0x67 => 0x00, // assuming behaviour is identical to that of 16-bit reads | ||||
|             0x68, 0x69 => @truncate(T, apu.ch2.sound2CntL() >> getHalf(byte_addr)), | ||||
|             0x6A, 0x6B => 0x00, | ||||
|             0x6C, 0x6D => @truncate(T, apu.ch2.sound2CntH() >> getHalf(byte_addr)), | ||||
|             0x6E, 0x6F => 0x00, | ||||
|             0x70, 0x71 => @truncate(T, @as(u16, apu.ch3.sound3CntL()) >> getHalf(byte_addr)), // SOUND3CNT_L | ||||
|             0x72, 0x73 => @truncate(T, apu.ch3.sound3CntH() >> getHalf(byte_addr)), | ||||
|             0x74, 0x75 => @truncate(T, apu.ch3.sound3CntX() >> getHalf(byte_addr)), // SOUND3CNT_L | ||||
|             0x76, 0x77 => 0x00, | ||||
|             0x78, 0x79 => @truncate(T, apu.ch4.sound4CntL() >> getHalf(byte_addr)), | ||||
|             0x7A, 0x7B => 0x00, | ||||
|             0x7C, 0x7D => @truncate(T, apu.ch4.sound4CntH() >> getHalf(byte_addr)), | ||||
|             0x7E, 0x7F => 0x00, | ||||
|             0x80, 0x81 => @truncate(T, apu.soundCntL() >> getHalf(byte_addr)), // SOUNDCNT_L | ||||
|             0x82, 0x83 => @truncate(T, apu.soundCntH() >> getHalf(byte_addr)), // SOUNDCNT_H | ||||
|             0x84, 0x85 => @truncate(T, @as(u16, apu.soundCntX()) >> getHalf(byte_addr)), | ||||
|             0x86, 0x87 => 0x00, | ||||
|             0x88, 0x89 => @truncate(T, apu.bias.raw >> getHalf(byte_addr)), // SOUNDBIAS | ||||
|             0x8A, 0x8B => 0x00, | ||||
|             0x8C...0x8F => null, | ||||
|             0x90...0x9F => apu.ch3.wave_dev.read(T, apu.ch3.select, addr), | ||||
|             0xA0, 0xA1, 0xA2, 0xA3 => null, // FIFO_A | ||||
|             0xA4, 0xA5, 0xA6, 0xA7 => null, // FIFO_B | ||||
|             else => util.io.read.err(T, log, "unexpected {} read from 0x{X:0>8}", .{ T, addr }), | ||||
|         }, | ||||
|         else => @compileError("APU: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(comptime T: type, apu: *Apu, addr: u32, value: T) void { | ||||
|     const byte_addr = @truncate(u8, addr); | ||||
|  | ||||
|     if (byte_addr <= 0x81 and !apu.cnt.apu_enable.read()) return; | ||||
|  | ||||
|     switch (T) { | ||||
|         u32 => { | ||||
|             // 0x80 and 0x81 handled in setSoundCnt | ||||
|             if (byte_addr < 0x80 and !apu.cnt.apu_enable.read()) return; | ||||
|  | ||||
|             switch (byte_addr) { | ||||
|                 0x60 => apu.ch1.setSound1Cnt(value), | ||||
|                 0x64 => apu.ch1.setSound1CntX(&apu.fs, @truncate(u16, value)), | ||||
|  | ||||
|                 0x68 => apu.ch2.setSound2CntL(@truncate(u16, value)), | ||||
|                 0x6C => apu.ch2.setSound2CntH(&apu.fs, @truncate(u16, value)), | ||||
|  | ||||
|                 0x70 => apu.ch3.setSound3Cnt(value), | ||||
|                 0x74 => apu.ch3.setSound3CntX(&apu.fs, @truncate(u16, value)), | ||||
|  | ||||
|                 0x78 => apu.ch4.setSound4CntL(@truncate(u16, value)), | ||||
|                 0x7C => apu.ch4.setSound4CntH(&apu.fs, @truncate(u16, value)), | ||||
|  | ||||
|                 0x80 => apu.setSoundCnt(value), | ||||
|                 0x84 => apu.setSoundCntX(value >> 7 & 1 == 1), | ||||
|                 0x88 => apu.bias.raw = @truncate(u16, value), | ||||
|                 0x8C => {}, | ||||
|  | ||||
|                 0x90, 0x94, 0x98, 0x9C => apu.ch3.wave_dev.write(T, apu.ch3.select, addr, value), | ||||
|                 0xA0 => apu.chA.push(value), // FIFO_A | ||||
|                 0xA4 => apu.chB.push(value), // FIFO_B | ||||
|                 else => util.io.write.undef(log, "Tried to write 0x{X:0>8}{} to 0x{X:0>8}", .{ value, T, addr }), | ||||
|             } | ||||
|         }, | ||||
|         u16 => { | ||||
|             if (byte_addr <= 0x81 and !apu.cnt.apu_enable.read()) return; | ||||
|  | ||||
|             switch (byte_addr) { | ||||
|                 0x60 => apu.ch1.setSound1CntL(@truncate(u8, value)), // SOUND1CNT_L | ||||
|                 0x62 => apu.ch1.setSound1CntH(value), | ||||
|                 0x64 => apu.ch1.setSound1CntX(&apu.fs, value), | ||||
|                 0x66 => {}, | ||||
|  | ||||
|                 0x68 => apu.ch2.setSound2CntL(value), | ||||
|                 0x6A => {}, | ||||
|                 0x6C => apu.ch2.setSound2CntH(&apu.fs, value), | ||||
|                 0x6E => {}, | ||||
|  | ||||
|                 0x70 => apu.ch3.setSound3CntL(@truncate(u8, value)), | ||||
|                 0x72 => apu.ch3.setSound3CntH(value), | ||||
|                 0x74 => apu.ch3.setSound3CntX(&apu.fs, value), | ||||
|                 0x76 => {}, | ||||
|  | ||||
|                 0x78 => apu.ch4.setSound4CntL(value), | ||||
|                 0x7A => {}, | ||||
|                 0x7C => apu.ch4.setSound4CntH(&apu.fs, value), | ||||
|                 0x7E => {}, | ||||
|  | ||||
|                 0x80 => apu.setSoundCntL(value), | ||||
|                 0x82 => apu.setSoundCntH(value), | ||||
|                 0x84 => apu.setSoundCntX(value >> 7 & 1 == 1), | ||||
|                 0x86 => {}, | ||||
|                 0x88 => apu.bias.raw = value, // SOUNDBIAS | ||||
|                 0x8A, 0x8C, 0x8E => {}, | ||||
|  | ||||
|                 0x90, 0x92, 0x94, 0x96, 0x98, 0x9A, 0x9C, 0x9E => apu.ch3.wave_dev.write(T, apu.ch3.select, addr, value), | ||||
|                 0xA0, 0xA2 => log.err("Tried to write 0x{X:0>4}{} to FIFO_A", .{ value, T }), | ||||
|                 0xA4, 0xA6 => log.err("Tried to write 0x{X:0>4}{} to FIFO_B", .{ value, T }), | ||||
|                 else => util.io.write.undef(log, "Tried to write 0x{X:0>4}{} to 0x{X:0>8}", .{ value, T, addr }), | ||||
|             } | ||||
|         }, | ||||
|         u8 => { | ||||
|             if (byte_addr <= 0x81 and !apu.cnt.apu_enable.read()) return; | ||||
|  | ||||
|             switch (byte_addr) { | ||||
|                 0x60 => apu.ch1.setSound1CntL(value), | ||||
|                 0x61 => {}, | ||||
|                 0x62 => apu.ch1.setNr11(value), | ||||
|                 0x63 => apu.ch1.setNr12(value), | ||||
|                 0x64 => apu.ch1.setNr13(value), | ||||
|                 0x65 => apu.ch1.setNr14(&apu.fs, value), | ||||
|                 0x66, 0x67 => {}, | ||||
|  | ||||
|                 0x68 => apu.ch2.setNr21(value), | ||||
|                 0x69 => apu.ch2.setNr22(value), | ||||
|                 0x6A, 0x6B => {}, | ||||
|                 0x6C => apu.ch2.setNr23(value), | ||||
|                 0x6D => apu.ch2.setNr24(&apu.fs, value), | ||||
|                 0x6E, 0x6F => {}, | ||||
|  | ||||
|                 0x70 => apu.ch3.setSound3CntL(value), // NR30 | ||||
|                 0x71 => {}, | ||||
|                 0x72 => apu.ch3.setNr31(value), | ||||
|                 0x73 => apu.ch3.vol.raw = value, // NR32 | ||||
|                 0x74 => apu.ch3.setNr33(value), | ||||
|                 0x75 => apu.ch3.setNr34(&apu.fs, value), | ||||
|                 0x76, 0x77 => {}, | ||||
|  | ||||
|                 0x78 => apu.ch4.setNr41(value), | ||||
|                 0x79 => apu.ch4.setNr42(value), | ||||
|                 0x7A, 0x7B => {}, | ||||
|                 0x7C => apu.ch4.poly.raw = value, // NR 43 | ||||
|                 0x7D => apu.ch4.setNr44(&apu.fs, value), | ||||
|                 0x7E, 0x7F => {}, | ||||
|  | ||||
|                 0x80, 0x81 => apu.setSoundCntL(setHalf(u16, apu.psg_cnt.raw, byte_addr, value)), | ||||
|                 0x82, 0x83 => apu.setSoundCntH(setHalf(u16, apu.dma_cnt.raw, byte_addr, value)), | ||||
|                 0x84 => apu.setSoundCntX(value >> 7 & 1 == 1), | ||||
|                 0x85 => {}, | ||||
|                 0x86, 0x87 => {}, | ||||
|                 0x88, 0x89 => apu.bias.raw = setHalf(u16, apu.bias.raw, byte_addr, value), // SOUNDBIAS | ||||
|                 0x8A...0x8F => {}, | ||||
|  | ||||
|                 0x90...0x9F => apu.ch3.wave_dev.write(T, apu.ch3.select, addr, value), | ||||
|                 0xA0...0xA3 => log.err("Tried to write 0x{X:0>2}{} to FIFO_A", .{ value, T }), | ||||
|                 0xA4...0xA7 => log.err("Tried to write 0x{X:0>2}{} to FIFO_B", .{ value, T }), | ||||
|                 else => util.io.write.undef(log, "Tried to write 0x{X:0>2}{} to 0x{X:0>8}", .{ value, T, addr }), | ||||
|             } | ||||
|         }, | ||||
|         else => @compileError("APU: Unsupported write width"), | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub const Apu = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     ch1: ToneSweep, | ||||
|     ch2: Tone, | ||||
|     ch3: Wave, | ||||
|     ch4: Noise, | ||||
|     chA: DmaSound(.A), | ||||
|     chB: DmaSound(.B), | ||||
|  | ||||
|     bias: io.SoundBias, | ||||
|     /// NR50, NR51 | ||||
|     psg_cnt: io.ChannelVolumeControl, | ||||
|     dma_cnt: io.DmaSoundControl, | ||||
|     cnt: io.SoundControl, | ||||
|  | ||||
|     sampling_cycle: u2, | ||||
|  | ||||
|     sample_queue: RingBuffer(u16), | ||||
|     sched: *Scheduler, | ||||
|  | ||||
|     fs: FrameSequencer, | ||||
|     capacitor: f32, | ||||
|  | ||||
|     pub const Tick = enum { Length, Envelope, Sweep }; | ||||
|  | ||||
|     pub fn init(sched: *Scheduler) Self { | ||||
|         const NUM_CHANNELS: usize = 2; | ||||
|  | ||||
|         const allocator = std.heap.c_allocator; | ||||
|         const sample_buf = allocator.alloc(u16, 0x800 * NUM_CHANNELS) catch @panic("failed to allocate sample buffer"); | ||||
|  | ||||
|         const apu: Self = .{ | ||||
|             .ch1 = ToneSweep.init(sched), | ||||
|             .ch2 = Tone.init(sched), | ||||
|             .ch3 = Wave.init(sched), | ||||
|             .ch4 = Noise.init(sched), | ||||
|             .chA = DmaSound(.A).init(), | ||||
|             .chB = DmaSound(.B).init(), | ||||
|  | ||||
|             .psg_cnt = .{ .raw = 0 }, | ||||
|             .dma_cnt = .{ .raw = 0 }, | ||||
|             .cnt = .{ .raw = 0 }, | ||||
|             .bias = .{ .raw = 0x0200 }, | ||||
|  | ||||
|             .sampling_cycle = 0b00, | ||||
|             .sample_queue = RingBuffer(u16).init(sample_buf), | ||||
|             .sched = sched, | ||||
|  | ||||
|             .capacitor = 0, | ||||
|             .fs = FrameSequencer.init(), | ||||
|         }; | ||||
|  | ||||
|         sched.push(.SampleAudio, apu.interval()); | ||||
|         sched.push(.{ .ApuChannel = 0 }, @import("apu/signal/Square.zig").interval); | ||||
|         sched.push(.{ .ApuChannel = 1 }, @import("apu/signal/Square.zig").interval); | ||||
|         sched.push(.{ .ApuChannel = 2 }, @import("apu/signal/Wave.zig").interval); | ||||
|         sched.push(.{ .ApuChannel = 3 }, @import("apu/signal/Lfsr.zig").interval); | ||||
|         sched.push(.FrameSequencer, FrameSequencer.interval); | ||||
|  | ||||
|         return apu; | ||||
|     } | ||||
|  | ||||
|     fn reset(self: *Self) void { | ||||
|         // All PSG Registers between 0x0400_0060..0x0400_0081 are zeroed | ||||
|         // 0x0400_0082 and 0x0400_0088 retain their values | ||||
|         self.ch1.reset(); | ||||
|         self.ch2.reset(); | ||||
|         self.ch3.reset(); | ||||
|         self.ch4.reset(); | ||||
|  | ||||
|         // GBATEK says 4000060h..4000081h I take this to mean inclusive | ||||
|         self.psg_cnt.raw = 0x0000; | ||||
|     } | ||||
|  | ||||
|     /// SOUNDCNT | ||||
|     fn setSoundCnt(self: *Self, value: u32) void { | ||||
|         if (self.cnt.apu_enable.read()) self.setSoundCntL(@truncate(u16, value)); | ||||
|         self.setSoundCntH(@truncate(u16, value >> 16)); | ||||
|     } | ||||
|  | ||||
|     /// SOUNDCNT_L | ||||
|     pub fn soundCntL(self: *const Self) u16 { | ||||
|         return self.psg_cnt.raw & 0xFF77; | ||||
|     } | ||||
|  | ||||
|     /// SOUNDCNT_L | ||||
|     pub fn setSoundCntL(self: *Self, value: u16) void { | ||||
|         self.psg_cnt.raw = value; | ||||
|     } | ||||
|  | ||||
|     /// SOUNDCNT_H | ||||
|     pub fn setSoundCntH(self: *Self, value: u16) void { | ||||
|         const new: io.DmaSoundControl = .{ .raw = value }; | ||||
|  | ||||
|         // Reinitializing instead of resetting is fine because | ||||
|         // the FIFOs I'm using are stack allocated and 0x20 bytes big | ||||
|         if (new.chA_reset.read()) self.chA.fifo = SoundFifo.init(); | ||||
|         if (new.chB_reset.read()) self.chB.fifo = SoundFifo.init(); | ||||
|  | ||||
|         self.dma_cnt = new; | ||||
|     } | ||||
|  | ||||
|     /// SOUNDCNT_H | ||||
|     pub fn soundCntH(self: *const Self) u16 { | ||||
|         return self.dma_cnt.raw & 0x770F; | ||||
|     } | ||||
|  | ||||
|     /// NR52 | ||||
|     pub fn setSoundCntX(self: *Self, value: bool) void { | ||||
|         self.cnt.apu_enable.write(value); | ||||
|  | ||||
|         if (value) { | ||||
|             self.fs.step = 0; // Reset Frame Sequencer | ||||
|  | ||||
|             // Reset Square Wave Offsets | ||||
|             self.ch1.square.reset(); | ||||
|             self.ch2.square.reset(); | ||||
|  | ||||
|             // Reset Wave | ||||
|             self.ch3.wave_dev.reset(); | ||||
|  | ||||
|             // Rest Noise | ||||
|             self.ch4.lfsr.reset(); | ||||
|         } else { | ||||
|             self.reset(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// NR52 | ||||
|     pub fn soundCntX(self: *const Self) u8 { | ||||
|         const apu_enable: u8 = @boolToInt(self.cnt.apu_enable.read()); | ||||
|  | ||||
|         const ch1_enable: u8 = @boolToInt(self.ch1.enabled); | ||||
|         const ch2_enable: u8 = @boolToInt(self.ch2.enabled); | ||||
|         const ch3_enable: u8 = @boolToInt(self.ch3.enabled); | ||||
|         const ch4_enable: u8 = @boolToInt(self.ch4.enabled); | ||||
|  | ||||
|         return apu_enable << 7 | ch4_enable << 3 | ch3_enable << 2 | ch2_enable << 1 | ch1_enable; | ||||
|     } | ||||
|  | ||||
|     pub fn sampleAudio(self: *Self, late: u64) void { | ||||
|         self.sched.push(.SampleAudio, self.interval() -| late); | ||||
|  | ||||
|         var left: i16 = 0; | ||||
|         var right: i16 = 0; | ||||
|  | ||||
|         // SOUNDCNT_L Channel Enable flags | ||||
|         const ch_left: u4 = self.psg_cnt.ch_left.read(); | ||||
|         const ch_right: u4 = self.psg_cnt.ch_right.read(); | ||||
|  | ||||
|         // Determine SOUNDCNT_H volume modifications | ||||
|         const gba_vol: u4 = switch (self.dma_cnt.ch_vol.read()) { | ||||
|             0b00 => 2, | ||||
|             0b01 => 1, | ||||
|             else => 0, | ||||
|         }; | ||||
|  | ||||
|         // Add all PSG channels together | ||||
|         left += if (ch_left & 1 == 1) @as(i16, self.ch1.sample) else 0; | ||||
|         left += if (ch_left >> 1 & 1 == 1) @as(i16, self.ch2.sample) else 0; | ||||
|         left += if (ch_left >> 2 & 1 == 1) @as(i16, self.ch3.sample) else 0; | ||||
|         left += if (ch_left >> 3 == 1) @as(i16, self.ch4.sample) else 0; | ||||
|  | ||||
|         right += if (ch_right & 1 == 1) @as(i16, self.ch1.sample) else 0; | ||||
|         right += if (ch_right >> 1 & 1 == 1) @as(i16, self.ch2.sample) else 0; | ||||
|         right += if (ch_right >> 2 & 1 == 1) @as(i16, self.ch3.sample) else 0; | ||||
|         right += if (ch_right >> 3 == 1) @as(i16, self.ch4.sample) else 0; | ||||
|  | ||||
|         // Multiply by master channel volume | ||||
|         left *= 1 + @as(i16, self.psg_cnt.left_vol.read()); | ||||
|         right *= 1 + @as(i16, self.psg_cnt.right_vol.read()); | ||||
|  | ||||
|         // Apply GBA volume modifications to PSG Channels | ||||
|         left >>= gba_vol; | ||||
|         right >>= gba_vol; | ||||
|  | ||||
|         const chA_sample = self.chA.amplitude() << if (self.dma_cnt.chA_vol.read()) @as(u4, 2) else 1; | ||||
|         const chB_sample = self.chB.amplitude() << if (self.dma_cnt.chB_vol.read()) @as(u4, 2) else 1; | ||||
|  | ||||
|         left += if (self.dma_cnt.chA_left.read()) chA_sample else 0; | ||||
|         left += if (self.dma_cnt.chB_left.read()) chB_sample else 0; | ||||
|  | ||||
|         right += if (self.dma_cnt.chA_right.read()) chA_sample else 0; | ||||
|         right += if (self.dma_cnt.chB_right.read()) chB_sample else 0; | ||||
|  | ||||
|         // Add SOUNDBIAS | ||||
|         // FIXME: SOUNDBIAS is 10-bit but The waveform is centered around 0 if I treat it as 11-bit | ||||
|         const bias = @as(i16, self.bias.level.read()) << 2; | ||||
|         left += bias; | ||||
|         right += bias; | ||||
|  | ||||
|         const clamped_left = std.math.clamp(@bitCast(u16, left), std.math.minInt(u11), std.math.maxInt(u11)); | ||||
|         const clamped_right = std.math.clamp(@bitCast(u16, right), std.math.minInt(u11), std.math.maxInt(u11)); | ||||
|  | ||||
|         // Extend to 16-bit signed audio samples | ||||
|         const ext_left = (clamped_left << 5) | (clamped_left >> 6); | ||||
|         const ext_right = (clamped_right << 5) | (clamped_right >> 6); | ||||
|  | ||||
|         self.sample_queue.push(ext_left, ext_right) catch {}; | ||||
|     } | ||||
|  | ||||
|     fn interval(self: *const Self) u64 { | ||||
|         return (1 << 24) / Self.sampleRate(self.bias.sampling_cycle.read()); | ||||
|     } | ||||
|  | ||||
|     fn sampleRate(cycle: u2) u64 { | ||||
|         return @as(u64, 1) << (15 + @as(u6, cycle)); | ||||
|     } | ||||
|  | ||||
|     pub fn onSequencerTick(self: *Self, late: u64) void { | ||||
|         self.fs.tick(); | ||||
|  | ||||
|         switch (self.fs.step) { | ||||
|             7 => self.tick(.Envelope), // Clock Envelope | ||||
|             0, 4 => self.tick(.Length), // Clock Length | ||||
|             2, 6 => { | ||||
|                 // Clock Length and Sweep | ||||
|                 self.tick(.Length); | ||||
|                 self.tick(.Sweep); | ||||
|             }, | ||||
|             1, 3, 5 => {}, | ||||
|         } | ||||
|  | ||||
|         self.sched.push(.FrameSequencer, ((1 << 24) / 512) -| late); | ||||
|     } | ||||
|  | ||||
|     fn tick(self: *Self, comptime kind: Tick) void { | ||||
|         self.ch1.tick(kind); | ||||
|  | ||||
|         switch (kind) { | ||||
|             .Length => { | ||||
|                 self.ch2.tick(kind); | ||||
|                 self.ch3.tick(kind); | ||||
|                 self.ch4.tick(kind); | ||||
|             }, | ||||
|             .Envelope => { | ||||
|                 self.ch2.tick(kind); | ||||
|                 self.ch4.tick(kind); | ||||
|             }, | ||||
|             .Sweep => {}, // Already handled above (only for Ch1) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn onDmaAudioSampleRequest(self: *Self, cpu: *Arm7tdmi, tim_id: u3) void { | ||||
|         if (!self.cnt.apu_enable.read()) return; | ||||
|  | ||||
|         if (@boolToInt(self.dma_cnt.chA_timer.read()) == tim_id) { | ||||
|             if (!self.chA.enabled) return; | ||||
|  | ||||
|             self.chA.updateSample(); | ||||
|             if (self.chA.len() <= 15) cpu.bus.dma[1].requestAudio(0x0400_00A0); | ||||
|         } | ||||
|  | ||||
|         if (@boolToInt(self.dma_cnt.chB_timer.read()) == tim_id) { | ||||
|             if (!self.chB.enabled) return; | ||||
|  | ||||
|             self.chB.updateSample(); | ||||
|             if (self.chB.len() <= 15) cpu.bus.dma[2].requestAudio(0x0400_00A4); | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| pub fn DmaSound(comptime kind: DmaSoundKind) type { | ||||
|     return struct { | ||||
|         const Self = @This(); | ||||
|  | ||||
|         fifo: SoundFifo, | ||||
|         kind: DmaSoundKind, | ||||
|         sample: i8, | ||||
|         enabled: bool, | ||||
|  | ||||
|         fn init() Self { | ||||
|             return .{ | ||||
|                 .fifo = SoundFifo.init(), | ||||
|                 .kind = kind, | ||||
|                 .sample = 0, | ||||
|                 .enabled = false, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         pub fn push(self: *Self, value: u32) void { | ||||
|             if (!self.enabled) self.enable(); | ||||
|  | ||||
|             self.fifo.write(&intToBytes(u32, value)) catch |e| log.err("{} Error: {}", .{ kind, e }); | ||||
|         } | ||||
|  | ||||
|         fn enable(self: *Self) void { | ||||
|             @setCold(true); | ||||
|             self.enabled = true; | ||||
|         } | ||||
|  | ||||
|         pub fn len(self: *const Self) usize { | ||||
|             return self.fifo.readableLength(); | ||||
|         } | ||||
|  | ||||
|         pub fn updateSample(self: *Self) void { | ||||
|             if (self.fifo.readItem()) |sample| self.sample = @bitCast(i8, sample); | ||||
|         } | ||||
|  | ||||
|         pub fn amplitude(self: *const Self) i16 { | ||||
|             return @as(i16, self.sample); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
|  | ||||
| const DmaSoundKind = enum { | ||||
|     A, | ||||
|     B, | ||||
| }; | ||||
|  | ||||
| pub const FrameSequencer = struct { | ||||
|     const Self = @This(); | ||||
|     pub const interval = (1 << 24) / 512; | ||||
|  | ||||
|     step: u3, | ||||
|  | ||||
|     pub fn init() Self { | ||||
|         return .{ .step = 0 }; | ||||
|     } | ||||
|  | ||||
|     pub fn tick(self: *Self) void { | ||||
|         self.step +%= 1; | ||||
|     } | ||||
|  | ||||
|     pub fn isLengthNext(self: *const Self) bool { | ||||
|         return (self.step +% 1) & 1 == 0; // Steps, 0, 2, 4, and 6 clock length | ||||
|     } | ||||
|  | ||||
|     pub fn isEnvelopeNext(self: *const Self) bool { | ||||
|         return (self.step +% 1) == 7; | ||||
|     } | ||||
| }; | ||||
| @@ -1,145 +0,0 @@ | ||||
| const io = @import("../bus/io.zig"); | ||||
| const util = @import("../../util.zig"); | ||||
|  | ||||
| const Scheduler = @import("../scheduler.zig").Scheduler; | ||||
| const FrameSequencer = @import("../apu.zig").FrameSequencer; | ||||
| const Tick = @import("../apu.zig").Apu.Tick; | ||||
| const Envelope = @import("device/Envelope.zig"); | ||||
| const Length = @import("device/Length.zig"); | ||||
| const Lfsr = @import("signal/Lfsr.zig"); | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| /// Write-only | ||||
| /// NR41 | ||||
| len: u6, | ||||
| /// NR42 | ||||
| envelope: io.Envelope, | ||||
| /// NR43 | ||||
| poly: io.PolyCounter, | ||||
| /// NR44 | ||||
| cnt: io.NoiseControl, | ||||
|  | ||||
| /// Length Functionarlity | ||||
| len_dev: Length, | ||||
|  | ||||
| /// Envelope Functionality | ||||
| env_dev: Envelope, | ||||
|  | ||||
| // Linear Feedback Shift Register | ||||
| lfsr: Lfsr, | ||||
|  | ||||
| enabled: bool, | ||||
| sample: i8, | ||||
|  | ||||
| pub fn init(sched: *Scheduler) Self { | ||||
|     return .{ | ||||
|         .len = 0, | ||||
|         .envelope = .{ .raw = 0 }, | ||||
|         .poly = .{ .raw = 0 }, | ||||
|         .cnt = .{ .raw = 0 }, | ||||
|         .enabled = false, | ||||
|  | ||||
|         .len_dev = Length.create(), | ||||
|         .env_dev = Envelope.create(), | ||||
|         .lfsr = Lfsr.create(sched), | ||||
|  | ||||
|         .sample = 0, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.len = 0; // NR41 | ||||
|     self.envelope.raw = 0; // NR42 | ||||
|     self.poly.raw = 0; // NR43 | ||||
|     self.cnt.raw = 0; // NR44 | ||||
|  | ||||
|     self.len_dev.reset(); | ||||
|     self.env_dev.reset(); | ||||
|  | ||||
|     self.sample = 0; | ||||
|     self.enabled = false; | ||||
| } | ||||
|  | ||||
| pub fn tick(self: *Self, comptime kind: Tick) void { | ||||
|     switch (kind) { | ||||
|         .Length => self.len_dev.tick(self.cnt.length_enable.read(), &self.enabled), | ||||
|         .Envelope => self.env_dev.tick(self.envelope), | ||||
|         .Sweep => @compileError("Channel 4 does not implement Sweep"), | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// NR41, NR42 | ||||
| pub fn sound4CntL(self: *const Self) u16 { | ||||
|     return @as(u16, self.envelope.raw) << 8; | ||||
| } | ||||
|  | ||||
| /// NR41, NR42 | ||||
| pub fn setSound4CntL(self: *Self, value: u16) void { | ||||
|     self.setNr41(@truncate(u8, value)); | ||||
|     self.setNr42(@truncate(u8, value >> 8)); | ||||
| } | ||||
|  | ||||
| /// NR41 | ||||
| pub fn setNr41(self: *Self, len: u8) void { | ||||
|     self.len = @truncate(u6, len); | ||||
|     self.len_dev.timer = @as(u7, 64) - @truncate(u6, len); | ||||
| } | ||||
|  | ||||
| /// NR42 | ||||
| pub fn setNr42(self: *Self, value: u8) void { | ||||
|     self.envelope.raw = value; | ||||
|     if (!self.isDacEnabled()) self.enabled = false; | ||||
| } | ||||
|  | ||||
| /// NR43, NR44 | ||||
| pub fn sound4CntH(self: *const Self) u16 { | ||||
|     return @as(u16, self.poly.raw & 0x40) << 8 | self.cnt.raw; | ||||
| } | ||||
|  | ||||
| /// NR43, NR44 | ||||
| pub fn setSound4CntH(self: *Self, fs: *const FrameSequencer, value: u16) void { | ||||
|     self.poly.raw = @truncate(u8, value); | ||||
|     self.setNr44(fs, @truncate(u8, value >> 8)); | ||||
| } | ||||
|  | ||||
| /// NR44 | ||||
| pub fn setNr44(self: *Self, fs: *const FrameSequencer, byte: u8) void { | ||||
|     var new: io.NoiseControl = .{ .raw = byte }; | ||||
|  | ||||
|     if (new.trigger.read()) { | ||||
|         self.enabled = true; | ||||
|  | ||||
|         if (self.len_dev.timer == 0) { | ||||
|             self.len_dev.timer = | ||||
|                 if (!fs.isLengthNext() and new.length_enable.read()) 63 else 64; | ||||
|         } | ||||
|  | ||||
|         // Update The Frequency Timer | ||||
|         self.lfsr.reload(self.poly); | ||||
|         self.lfsr.shift = 0x7FFF; | ||||
|  | ||||
|         // Update Envelope and Volume | ||||
|         self.env_dev.timer = self.envelope.period.read(); | ||||
|         if (fs.isEnvelopeNext() and self.env_dev.timer != 0b111) self.env_dev.timer += 1; | ||||
|  | ||||
|         self.env_dev.vol = self.envelope.init_vol.read(); | ||||
|  | ||||
|         self.enabled = self.isDacEnabled(); | ||||
|     } | ||||
|  | ||||
|     util.audio.length.ch4.update(self, fs, new); | ||||
|     self.cnt = new; | ||||
| } | ||||
|  | ||||
| pub fn onNoiseEvent(self: *Self, late: u64) void { | ||||
|     self.lfsr.onLfsrTimerExpire(self.poly, late); | ||||
|  | ||||
|     self.sample = 0; | ||||
|     if (!self.isDacEnabled()) return; | ||||
|     self.sample = if (self.enabled) self.lfsr.sample() * @as(i8, self.env_dev.vol) else 0; | ||||
| } | ||||
|  | ||||
| fn isDacEnabled(self: *const Self) bool { | ||||
|     return self.envelope.raw & 0xF8 != 0x00; | ||||
| } | ||||
| @@ -1,141 +0,0 @@ | ||||
| const io = @import("../bus/io.zig"); | ||||
| const util = @import("../../util.zig"); | ||||
|  | ||||
| const Scheduler = @import("../scheduler.zig").Scheduler; | ||||
| const FrameSequencer = @import("../apu.zig").FrameSequencer; | ||||
| const Tick = @import("../apu.zig").Apu.Tick; | ||||
| const Length = @import("device/Length.zig"); | ||||
| const Envelope = @import("device/Envelope.zig"); | ||||
| const Square = @import("signal/Square.zig"); | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| /// NR21 | ||||
| duty: io.Duty, | ||||
| /// NR22 | ||||
| envelope: io.Envelope, | ||||
| /// NR23, NR24 | ||||
| freq: io.Frequency, | ||||
|  | ||||
| /// Length Functionarlity | ||||
| len_dev: Length, | ||||
| /// Envelope Functionality | ||||
| env_dev: Envelope, | ||||
| /// FrequencyTimer Functionality | ||||
| square: Square, | ||||
|  | ||||
| enabled: bool, | ||||
| sample: i8, | ||||
|  | ||||
| pub fn init(sched: *Scheduler) Self { | ||||
|     return .{ | ||||
|         .duty = .{ .raw = 0 }, | ||||
|         .envelope = .{ .raw = 0 }, | ||||
|         .freq = .{ .raw = 0 }, | ||||
|         .enabled = false, | ||||
|  | ||||
|         .square = Square.init(sched), | ||||
|         .len_dev = Length.create(), | ||||
|         .env_dev = Envelope.create(), | ||||
|  | ||||
|         .sample = 0, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.duty.raw = 0; // NR21 | ||||
|     self.envelope.raw = 0; // NR22 | ||||
|     self.freq.raw = 0; // NR32, NR24 | ||||
|  | ||||
|     self.len_dev.reset(); | ||||
|     self.env_dev.reset(); | ||||
|  | ||||
|     self.sample = 0; | ||||
|     self.enabled = false; | ||||
| } | ||||
|  | ||||
| pub fn tick(self: *Self, comptime kind: Tick) void { | ||||
|     switch (kind) { | ||||
|         .Length => self.len_dev.tick(self.freq.length_enable.read(), &self.enabled), | ||||
|         .Envelope => self.env_dev.tick(self.envelope), | ||||
|         .Sweep => @compileError("Channel 2 does not implement Sweep"), | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn onToneEvent(self: *Self, late: u64) void { | ||||
|     self.square.onSquareTimerExpire(Self, self.freq, late); | ||||
|  | ||||
|     self.sample = 0; | ||||
|     if (!self.isDacEnabled()) return; | ||||
|     self.sample = if (self.enabled) self.square.sample(self.duty) * @as(i8, self.env_dev.vol) else 0; | ||||
| } | ||||
|  | ||||
| /// NR21, NR22 | ||||
| pub fn sound2CntL(self: *const Self) u16 { | ||||
|     return @as(u16, self.envelope.raw) << 8 | (self.duty.raw & 0xC0); | ||||
| } | ||||
|  | ||||
| /// NR21, NR22 | ||||
| pub fn setSound2CntL(self: *Self, value: u16) void { | ||||
|     self.setNr21(@truncate(u8, value)); | ||||
|     self.setNr22(@truncate(u8, value >> 8)); | ||||
| } | ||||
|  | ||||
| /// NR21 | ||||
| pub fn setNr21(self: *Self, value: u8) void { | ||||
|     self.duty.raw = value; | ||||
|     self.len_dev.timer = @as(u7, 64) - @truncate(u6, value); | ||||
| } | ||||
|  | ||||
| /// NR22 | ||||
| pub fn setNr22(self: *Self, value: u8) void { | ||||
|     self.envelope.raw = value; | ||||
|     if (!self.isDacEnabled()) self.enabled = false; | ||||
| } | ||||
|  | ||||
| /// NR23, NR24 | ||||
| pub fn sound2CntH(self: *const Self) u16 { | ||||
|     return self.freq.raw & 0x4000; | ||||
| } | ||||
|  | ||||
| /// NR23, NR24 | ||||
| pub fn setSound2CntH(self: *Self, fs: *const FrameSequencer, value: u16) void { | ||||
|     self.setNr23(@truncate(u8, value)); | ||||
|     self.setNr24(fs, @truncate(u8, value >> 8)); | ||||
| } | ||||
|  | ||||
| /// NR23 | ||||
| pub fn setNr23(self: *Self, byte: u8) void { | ||||
|     self.freq.raw = (self.freq.raw & 0xFF00) | byte; | ||||
| } | ||||
|  | ||||
| /// NR24 | ||||
| pub fn setNr24(self: *Self, fs: *const FrameSequencer, byte: u8) void { | ||||
|     var new: io.Frequency = .{ .raw = (@as(u16, byte) << 8) | (self.freq.raw & 0xFF) }; | ||||
|  | ||||
|     if (new.trigger.read()) { | ||||
|         self.enabled = true; | ||||
|  | ||||
|         if (self.len_dev.timer == 0) { | ||||
|             self.len_dev.timer = | ||||
|                 if (!fs.isLengthNext() and new.length_enable.read()) 63 else 64; | ||||
|         } | ||||
|  | ||||
|         self.square.reload(Self, self.freq.frequency.read()); | ||||
|  | ||||
|         // Reload Envelope period and timer | ||||
|         self.env_dev.timer = self.envelope.period.read(); | ||||
|         if (fs.isEnvelopeNext() and self.env_dev.timer != 0b111) self.env_dev.timer += 1; | ||||
|  | ||||
|         self.env_dev.vol = self.envelope.init_vol.read(); | ||||
|  | ||||
|         self.enabled = self.isDacEnabled(); | ||||
|     } | ||||
|  | ||||
|     util.audio.length.update(Self, self, fs, new); | ||||
|     self.freq = new; | ||||
| } | ||||
|  | ||||
| fn isDacEnabled(self: *const Self) bool { | ||||
|     return self.envelope.raw & 0xF8 != 0; | ||||
| } | ||||
| @@ -1,185 +0,0 @@ | ||||
| const io = @import("../bus/io.zig"); | ||||
| const util = @import("../../util.zig"); | ||||
|  | ||||
| const Scheduler = @import("../scheduler.zig").Scheduler; | ||||
| const FrameSequencer = @import("../apu.zig").FrameSequencer; | ||||
| const Length = @import("device/Length.zig"); | ||||
| const Envelope = @import("device/Envelope.zig"); | ||||
| const Sweep = @import("device/Sweep.zig"); | ||||
| const Square = @import("signal/Square.zig"); | ||||
|  | ||||
| const Tick = @import("../apu.zig").Apu.Tick; | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| /// NR10 | ||||
| sweep: io.Sweep, | ||||
| /// NR11 | ||||
| duty: io.Duty, | ||||
| /// NR12 | ||||
| envelope: io.Envelope, | ||||
| /// NR13, NR14 | ||||
| freq: io.Frequency, | ||||
|  | ||||
| /// Length Functionality | ||||
| len_dev: Length, | ||||
| /// Sweep Functionality | ||||
| sweep_dev: Sweep, | ||||
| /// Envelope Functionality | ||||
| env_dev: Envelope, | ||||
| /// Frequency Timer Functionality | ||||
| square: Square, | ||||
| enabled: bool, | ||||
|  | ||||
| sample: i8, | ||||
|  | ||||
| pub fn init(sched: *Scheduler) Self { | ||||
|     return .{ | ||||
|         .sweep = .{ .raw = 0 }, | ||||
|         .duty = .{ .raw = 0 }, | ||||
|         .envelope = .{ .raw = 0 }, | ||||
|         .freq = .{ .raw = 0 }, | ||||
|         .sample = 0, | ||||
|         .enabled = false, | ||||
|  | ||||
|         .square = Square.init(sched), | ||||
|         .len_dev = Length.create(), | ||||
|         .sweep_dev = Sweep.create(), | ||||
|         .env_dev = Envelope.create(), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.sweep.raw = 0; // NR10 | ||||
|     self.duty.raw = 0; // NR11 | ||||
|     self.envelope.raw = 0; // NR12 | ||||
|     self.freq.raw = 0; // NR13, NR14 | ||||
|  | ||||
|     self.len_dev.reset(); | ||||
|     self.sweep_dev.reset(); | ||||
|     self.env_dev.reset(); | ||||
|  | ||||
|     self.sample = 0; | ||||
|     self.enabled = false; | ||||
| } | ||||
|  | ||||
| pub fn tick(self: *Self, comptime kind: Tick) void { | ||||
|     switch (kind) { | ||||
|         .Length => self.len_dev.tick(self.freq.length_enable.read(), &self.enabled), | ||||
|         .Envelope => self.env_dev.tick(self.envelope), | ||||
|         .Sweep => self.sweep_dev.tick(self), | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn onToneSweepEvent(self: *Self, late: u64) void { | ||||
|     self.square.onSquareTimerExpire(Self, self.freq, late); | ||||
|  | ||||
|     self.sample = 0; | ||||
|     if (!self.isDacEnabled()) return; | ||||
|     self.sample = if (self.enabled) self.square.sample(self.duty) * @as(i8, self.env_dev.vol) else 0; | ||||
| } | ||||
|  | ||||
| /// NR10, NR11, NR12 | ||||
| pub fn setSound1Cnt(self: *Self, value: u32) void { | ||||
|     self.setSound1CntL(@truncate(u8, value)); | ||||
|     self.setSound1CntH(@truncate(u16, value >> 16)); | ||||
| } | ||||
|  | ||||
| /// NR10 | ||||
| pub fn sound1CntL(self: *const Self) u8 { | ||||
|     return self.sweep.raw & 0x7F; | ||||
| } | ||||
|  | ||||
| /// NR10 | ||||
| pub fn setSound1CntL(self: *Self, value: u8) void { | ||||
|     const new = io.Sweep{ .raw = value }; | ||||
|  | ||||
|     if (!new.direction.read()) { | ||||
|         // If at least one (1) sweep calculation has been made with | ||||
|         // the negate bit set (since last trigger), disable the channel | ||||
|  | ||||
|         if (self.sweep_dev.calc_performed) self.enabled = false; | ||||
|     } | ||||
|  | ||||
|     self.sweep.raw = value; | ||||
| } | ||||
|  | ||||
| /// NR11, NR12 | ||||
| pub fn sound1CntH(self: *const Self) u16 { | ||||
|     return @as(u16, self.envelope.raw) << 8 | (self.duty.raw & 0xC0); | ||||
| } | ||||
|  | ||||
| /// NR11, NR12 | ||||
| pub fn setSound1CntH(self: *Self, value: u16) void { | ||||
|     self.setNr11(@truncate(u8, value)); | ||||
|     self.setNr12(@truncate(u8, value >> 8)); | ||||
| } | ||||
|  | ||||
| /// NR11 | ||||
| pub fn setNr11(self: *Self, value: u8) void { | ||||
|     self.duty.raw = value; | ||||
|     self.len_dev.timer = @as(u7, 64) - @truncate(u6, value); | ||||
| } | ||||
|  | ||||
| /// NR12 | ||||
| pub fn setNr12(self: *Self, value: u8) void { | ||||
|     self.envelope.raw = value; | ||||
|     if (!self.isDacEnabled()) self.enabled = false; | ||||
| } | ||||
|  | ||||
| /// NR13, NR14 | ||||
| pub fn sound1CntX(self: *const Self) u16 { | ||||
|     return self.freq.raw & 0x4000; | ||||
| } | ||||
|  | ||||
| /// NR13, NR14 | ||||
| pub fn setSound1CntX(self: *Self, fs: *const FrameSequencer, value: u16) void { | ||||
|     self.setNr13(@truncate(u8, value)); | ||||
|     self.setNr14(fs, @truncate(u8, value >> 8)); | ||||
| } | ||||
|  | ||||
| /// NR13 | ||||
| pub fn setNr13(self: *Self, byte: u8) void { | ||||
|     self.freq.raw = (self.freq.raw & 0xFF00) | byte; | ||||
| } | ||||
|  | ||||
| /// NR14 | ||||
| pub fn setNr14(self: *Self, fs: *const FrameSequencer, byte: u8) void { | ||||
|     var new: io.Frequency = .{ .raw = (@as(u16, byte) << 8) | (self.freq.raw & 0xFF) }; | ||||
|  | ||||
|     if (new.trigger.read()) { | ||||
|         self.enabled = true; | ||||
|  | ||||
|         if (self.len_dev.timer == 0) { | ||||
|             self.len_dev.timer = | ||||
|                 if (!fs.isLengthNext() and new.length_enable.read()) 63 else 64; | ||||
|         } | ||||
|  | ||||
|         self.square.reload(Self, self.freq.frequency.read()); | ||||
|  | ||||
|         // Reload Envelope period and timer | ||||
|         self.env_dev.timer = self.envelope.period.read(); | ||||
|         if (fs.isEnvelopeNext() and self.env_dev.timer != 0b111) self.env_dev.timer += 1; | ||||
|  | ||||
|         self.env_dev.vol = self.envelope.init_vol.read(); | ||||
|  | ||||
|         // Sweep Trigger Behaviour | ||||
|         const sw_period = self.sweep.period.read(); | ||||
|         const sw_shift = self.sweep.shift.read(); | ||||
|  | ||||
|         self.sweep_dev.calc_performed = false; | ||||
|         self.sweep_dev.shadow = self.freq.frequency.read(); | ||||
|         self.sweep_dev.timer = if (sw_period == 0) 8 else sw_period; | ||||
|         self.sweep_dev.enabled = sw_period != 0 or sw_shift != 0; | ||||
|         if (sw_shift != 0) _ = self.sweep_dev.calculate(self.sweep, &self.enabled); | ||||
|  | ||||
|         self.enabled = self.isDacEnabled(); | ||||
|     } | ||||
|  | ||||
|     util.audio.length.update(Self, self, fs, new); | ||||
|     self.freq = new; | ||||
| } | ||||
|  | ||||
| fn isDacEnabled(self: *const Self) bool { | ||||
|     return self.envelope.raw & 0xF8 != 0; | ||||
| } | ||||
| @@ -1,145 +0,0 @@ | ||||
| const io = @import("../bus/io.zig"); | ||||
| const util = @import("../../util.zig"); | ||||
|  | ||||
| const Scheduler = @import("../scheduler.zig").Scheduler; | ||||
| const FrameSequencer = @import("../apu.zig").FrameSequencer; | ||||
| const Tick = @import("../apu.zig").Apu.Tick; | ||||
|  | ||||
| const Length = @import("device/Length.zig"); | ||||
| const Wave = @import("signal/Wave.zig"); | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| /// Write-only | ||||
| /// NR30 | ||||
| select: io.WaveSelect, | ||||
| /// NR31 | ||||
| length: u8, | ||||
| /// NR32 | ||||
| vol: io.WaveVolume, | ||||
| /// NR33, NR34 | ||||
| freq: io.Frequency, | ||||
|  | ||||
| /// Length Functionarlity | ||||
| len_dev: Length, | ||||
| wave_dev: Wave, | ||||
|  | ||||
| enabled: bool, | ||||
| sample: i8, | ||||
|  | ||||
| pub fn init(sched: *Scheduler) Self { | ||||
|     return .{ | ||||
|         .select = .{ .raw = 0 }, | ||||
|         .vol = .{ .raw = 0 }, | ||||
|         .freq = .{ .raw = 0 }, | ||||
|         .length = 0, | ||||
|  | ||||
|         .len_dev = Length.create(), | ||||
|         .wave_dev = Wave.init(sched), | ||||
|         .enabled = false, | ||||
|         .sample = 0, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.select.raw = 0; // NR30 | ||||
|     self.length = 0; // NR31 | ||||
|     self.vol.raw = 0; // NR32 | ||||
|     self.freq.raw = 0; // NR33, NR34 | ||||
|  | ||||
|     self.len_dev.reset(); | ||||
|     self.wave_dev.reset(); | ||||
|  | ||||
|     self.sample = 0; | ||||
|     self.enabled = false; | ||||
| } | ||||
|  | ||||
| pub fn tick(self: *Self, comptime kind: Tick) void { | ||||
|     switch (kind) { | ||||
|         .Length => self.len_dev.tick(self.freq.length_enable.read(), &self.enabled), | ||||
|         .Envelope => @compileError("Channel 3 does not implement Envelope"), | ||||
|         .Sweep => @compileError("Channel 3 does not implement Sweep"), | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// NR30, NR31, NR32 | ||||
| pub fn setSound3Cnt(self: *Self, value: u32) void { | ||||
|     self.setSound3CntL(@truncate(u8, value)); | ||||
|     self.setSound3CntH(@truncate(u16, value >> 16)); | ||||
| } | ||||
|  | ||||
| /// NR30 | ||||
| pub fn setSound3CntL(self: *Self, value: u8) void { | ||||
|     self.select.raw = value; | ||||
|     if (!self.select.enabled.read()) self.enabled = false; | ||||
| } | ||||
|  | ||||
| /// NR30 | ||||
| pub fn sound3CntL(self: *const Self) u8 { | ||||
|     return self.select.raw & 0xE0; | ||||
| } | ||||
|  | ||||
| /// NR31, NR32 | ||||
| pub fn sound3CntH(self: *const Self) u16 { | ||||
|     return @as(u16, self.length & 0xE0) << 8; | ||||
| } | ||||
|  | ||||
| /// NR31, NR32 | ||||
| pub fn setSound3CntH(self: *Self, value: u16) void { | ||||
|     self.setNr31(@truncate(u8, value)); | ||||
|     self.vol.raw = (@truncate(u8, value >> 8)); | ||||
| } | ||||
|  | ||||
| /// NR31 | ||||
| pub fn setNr31(self: *Self, len: u8) void { | ||||
|     self.length = len; | ||||
|     self.len_dev.timer = 256 - @as(u9, len); | ||||
| } | ||||
|  | ||||
| /// NR33, NR34 | ||||
| pub fn setSound3CntX(self: *Self, fs: *const FrameSequencer, value: u16) void { | ||||
|     self.setNr33(@truncate(u8, value)); | ||||
|     self.setNr34(fs, @truncate(u8, value >> 8)); | ||||
| } | ||||
|  | ||||
| /// NR33, NR34 | ||||
| pub fn sound3CntX(self: *const Self) u16 { | ||||
|     return self.freq.raw & 0x4000; | ||||
| } | ||||
|  | ||||
| /// NR33 | ||||
| pub fn setNr33(self: *Self, byte: u8) void { | ||||
|     self.freq.raw = (self.freq.raw & 0xFF00) | byte; | ||||
| } | ||||
|  | ||||
| /// NR34 | ||||
| pub fn setNr34(self: *Self, fs: *const FrameSequencer, byte: u8) void { | ||||
|     var new: io.Frequency = .{ .raw = (@as(u16, byte) << 8) | (self.freq.raw & 0xFF) }; | ||||
|  | ||||
|     if (new.trigger.read()) { | ||||
|         self.enabled = true; | ||||
|  | ||||
|         if (self.len_dev.timer == 0) { | ||||
|             self.len_dev.timer = | ||||
|                 if (!fs.isLengthNext() and new.length_enable.read()) 255 else 256; | ||||
|         } | ||||
|  | ||||
|         // Update The Frequency Timer | ||||
|         self.wave_dev.reload(self.freq.frequency.read()); | ||||
|         self.wave_dev.offset = 0; | ||||
|  | ||||
|         self.enabled = self.select.enabled.read(); | ||||
|     } | ||||
|  | ||||
|     util.audio.length.update(Self, self, fs, new); | ||||
|     self.freq = new; | ||||
| } | ||||
|  | ||||
| pub fn onWaveEvent(self: *Self, late: u64) void { | ||||
|     self.wave_dev.onWaveTimerExpire(self.freq, self.select, late); | ||||
|  | ||||
|     self.sample = 0; | ||||
|     if (!self.select.enabled.read()) return; | ||||
|     // Convert unsigned 4-bit wave sample to signed 8-bit sample | ||||
|     self.sample = (2 * @as(i8, self.wave_dev.sample(self.select)) - 15) >> self.wave_dev.shift(self.vol); | ||||
| } | ||||
| @@ -1,33 +0,0 @@ | ||||
| const io = @import("../../bus/io.zig"); | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| /// Period Timer | ||||
| timer: u3, | ||||
| /// Current Volume | ||||
| vol: u4, | ||||
|  | ||||
| pub fn create() Self { | ||||
|     return .{ .timer = 0, .vol = 0 }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.timer = 0; | ||||
|     self.vol = 0; | ||||
| } | ||||
|  | ||||
| pub fn tick(self: *Self, nrx2: io.Envelope) void { | ||||
|     if (nrx2.period.read() != 0) { | ||||
|         if (self.timer != 0) self.timer -= 1; | ||||
|  | ||||
|         if (self.timer == 0) { | ||||
|             self.timer = nrx2.period.read(); | ||||
|  | ||||
|             if (nrx2.direction.read()) { | ||||
|                 if (self.vol < 0xF) self.vol += 1; | ||||
|             } else { | ||||
|                 if (self.vol > 0x0) self.vol -= 1; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,22 +0,0 @@ | ||||
| const Self = @This(); | ||||
|  | ||||
| timer: u9, | ||||
|  | ||||
| pub fn create() Self { | ||||
|     return .{ .timer = 0 }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.timer = 0; | ||||
| } | ||||
|  | ||||
| pub fn tick(self: *Self, enabled: bool, ch_enable: *bool) void { | ||||
|     if (enabled) { | ||||
|         if (self.timer == 0) return; | ||||
|         self.timer -= 1; | ||||
|  | ||||
|         // By returning early if timer == 0, this is only | ||||
|         // true if timer == 0 because of the decrement we just did | ||||
|         if (self.timer == 0) ch_enable.* = false; | ||||
|     } | ||||
| } | ||||
| @@ -1,61 +0,0 @@ | ||||
| const io = @import("../../bus/io.zig"); | ||||
| const ToneSweep = @import("../ToneSweep.zig"); | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| timer: u8, | ||||
| enabled: bool, | ||||
| shadow: u11, | ||||
|  | ||||
| calc_performed: bool, | ||||
|  | ||||
| pub fn create() Self { | ||||
|     return .{ | ||||
|         .timer = 0, | ||||
|         .enabled = false, | ||||
|         .shadow = 0, | ||||
|         .calc_performed = false, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.timer = 0; | ||||
|     self.enabled = false; | ||||
|     self.shadow = 0; | ||||
|     self.calc_performed = false; | ||||
| } | ||||
|  | ||||
| pub fn tick(self: *Self, ch1: *ToneSweep) void { | ||||
|     if (self.timer != 0) self.timer -= 1; | ||||
|  | ||||
|     if (self.timer == 0) { | ||||
|         const period = ch1.sweep.period.read(); | ||||
|         self.timer = if (period == 0) 8 else period; | ||||
|  | ||||
|         if (self.enabled and period != 0) { | ||||
|             const new_freq = self.calculate(ch1.sweep, &ch1.enabled); | ||||
|  | ||||
|             if (new_freq <= 0x7FF and ch1.sweep.shift.read() != 0) { | ||||
|                 ch1.freq.frequency.write(@truncate(u11, new_freq)); | ||||
|                 self.shadow = @truncate(u11, new_freq); | ||||
|  | ||||
|                 _ = self.calculate(ch1.sweep, &ch1.enabled); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Calculates the Sweep Frequency | ||||
| pub fn calculate(self: *Self, sweep: io.Sweep, ch_enable: *bool) u12 { | ||||
|     const shadow = @as(u12, self.shadow); | ||||
|     const shadow_shifted = shadow >> sweep.shift.read(); | ||||
|     const decrease = sweep.direction.read(); | ||||
|  | ||||
|     const freq = if (decrease) blk: { | ||||
|         self.calc_performed = true; | ||||
|         break :blk shadow - shadow_shifted; | ||||
|     } else shadow + shadow_shifted; | ||||
|     if (freq > 0x7FF) ch_enable.* = false; | ||||
|  | ||||
|     return freq; | ||||
| } | ||||
| @@ -1,62 +0,0 @@ | ||||
| //! Linear Feedback Shift Register | ||||
| const io = @import("../../bus/io.zig"); | ||||
|  | ||||
| const Scheduler = @import("../../scheduler.zig").Scheduler; | ||||
|  | ||||
| const Self = @This(); | ||||
| pub const interval: u64 = (1 << 24) / (1 << 22); | ||||
|  | ||||
| shift: u15, | ||||
| timer: u16, | ||||
|  | ||||
| sched: *Scheduler, | ||||
|  | ||||
| pub fn create(sched: *Scheduler) Self { | ||||
|     return .{ | ||||
|         .shift = 0, | ||||
|         .timer = 0, | ||||
|         .sched = sched, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.shift = 0; | ||||
|     self.timer = 0; | ||||
| } | ||||
|  | ||||
| pub fn sample(self: *const Self) i8 { | ||||
|     return if ((~self.shift & 1) == 1) 1 else -1; | ||||
| } | ||||
|  | ||||
| /// Reload LFSR Timer | ||||
| pub fn reload(self: *Self, poly: io.PolyCounter) void { | ||||
|     self.sched.removeScheduledEvent(.{ .ApuChannel = 3 }); | ||||
|  | ||||
|     const div = Self.divisor(poly.div_ratio.read()); | ||||
|     const timer = div << poly.shift.read(); | ||||
|     self.sched.push(.{ .ApuChannel = 3 }, @as(u64, timer) * interval); | ||||
| } | ||||
|  | ||||
| /// Scheduler Event Handler for LFSR Timer Expire | ||||
| /// FIXME: This gets called a lot, slowing down the scheduler | ||||
| pub fn onLfsrTimerExpire(self: *Self, poly: io.PolyCounter, late: u64) void { | ||||
|     // Obscure: "Using a noise channel clock shift of 14 or 15 | ||||
|     // results in the LFSR receiving no clocks." | ||||
|     if (poly.shift.read() >= 14) return; | ||||
|  | ||||
|     const div = Self.divisor(poly.div_ratio.read()); | ||||
|     const timer = div << poly.shift.read(); | ||||
|  | ||||
|     const tmp = (self.shift & 1) ^ ((self.shift & 2) >> 1); | ||||
|     self.shift = (self.shift >> 1) | (tmp << 14); | ||||
|  | ||||
|     if (poly.width.read()) | ||||
|         self.shift = (self.shift & ~@as(u15, 0x40)) | tmp << 6; | ||||
|  | ||||
|     self.sched.push(.{ .ApuChannel = 3 }, @as(u64, timer) * interval -| late); | ||||
| } | ||||
|  | ||||
| fn divisor(code: u3) u16 { | ||||
|     if (code == 0) return 8; | ||||
|     return @as(u16, code) << 4; | ||||
| } | ||||
| @@ -1,62 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const io = @import("../../bus/io.zig"); | ||||
|  | ||||
| const Scheduler = @import("../../scheduler.zig").Scheduler; | ||||
| const ToneSweep = @import("../ToneSweep.zig"); | ||||
| const Tone = @import("../Tone.zig"); | ||||
|  | ||||
| const Self = @This(); | ||||
| pub const interval: u64 = (1 << 24) / (1 << 22); | ||||
|  | ||||
| pos: u3, | ||||
| sched: *Scheduler, | ||||
| timer: u16, | ||||
|  | ||||
| pub fn init(sched: *Scheduler) Self { | ||||
|     return .{ | ||||
|         .timer = 0, | ||||
|         .pos = 0, | ||||
|         .sched = sched, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.timer = 0; | ||||
|     self.pos = 0; | ||||
| } | ||||
|  | ||||
| /// Scheduler Event Handler for Square Synth Timer Expire | ||||
| pub fn onSquareTimerExpire(self: *Self, comptime T: type, nrx34: io.Frequency, late: u64) void { | ||||
|     comptime std.debug.assert(T == ToneSweep or T == Tone); | ||||
|     self.pos +%= 1; | ||||
|  | ||||
|     self.timer = (@as(u16, 2048) - nrx34.frequency.read()) * 4; | ||||
|     self.sched.push(.{ .ApuChannel = if (T == ToneSweep) 0 else 1 }, @as(u64, self.timer) * interval -| late); | ||||
| } | ||||
|  | ||||
| /// Reload Square Wave Timer | ||||
| pub fn reload(self: *Self, comptime T: type, value: u11) void { | ||||
|     comptime std.debug.assert(T == ToneSweep or T == Tone); | ||||
|     const channel = if (T == ToneSweep) 0 else 1; | ||||
|  | ||||
|     self.sched.removeScheduledEvent(.{ .ApuChannel = channel }); | ||||
|  | ||||
|     const tmp = (@as(u16, 2048) - value) * 4; // What Freq Timer should be assuming no weird behaviour | ||||
|     self.timer = (tmp & ~@as(u16, 0x3)) | self.timer & 0x3; // Keep the last two bits from the old timer; | ||||
|  | ||||
|     self.sched.push(.{ .ApuChannel = channel }, @as(u64, self.timer) * interval); | ||||
| } | ||||
|  | ||||
| pub fn sample(self: *const Self, nrx1: io.Duty) i8 { | ||||
|     const pattern = nrx1.pattern.read(); | ||||
|  | ||||
|     const i = self.pos ^ 7; // index of 0 should get highest bit | ||||
|     const result = switch (pattern) { | ||||
|         0b00 => @as(u8, 0b00000001) >> i, // 12.5% | ||||
|         0b01 => @as(u8, 0b00000011) >> i, // 25% | ||||
|         0b10 => @as(u8, 0b00001111) >> i, // 50% | ||||
|         0b11 => @as(u8, 0b11111100) >> i, // 75% | ||||
|     }; | ||||
|  | ||||
|     return if (result & 1 == 1) 1 else -1; | ||||
| } | ||||
| @@ -1,84 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const io = @import("../../bus/io.zig"); | ||||
|  | ||||
| const Scheduler = @import("../../scheduler.zig").Scheduler; | ||||
|  | ||||
| const buf_len = 0x20; | ||||
| pub const interval: u64 = (1 << 24) / (1 << 22); | ||||
| const Self = @This(); | ||||
|  | ||||
| buf: [buf_len]u8, | ||||
| timer: u16, | ||||
| offset: u12, | ||||
|  | ||||
| sched: *Scheduler, | ||||
|  | ||||
| pub fn read(self: *const Self, comptime T: type, nr30: io.WaveSelect, addr: u32) T { | ||||
|     // TODO: Handle reads when Channel 3 is disabled | ||||
|     const base = if (!nr30.bank.read()) @as(u32, 0x10) else 0; // Read from the Opposite Bank in Use | ||||
|  | ||||
|     const i = base + addr - 0x0400_0090; | ||||
|     return std.mem.readIntSliceLittle(T, self.buf[i..][0..@sizeOf(T)]); | ||||
| } | ||||
|  | ||||
| pub fn write(self: *Self, comptime T: type, nr30: io.WaveSelect, addr: u32, value: T) void { | ||||
|     // TODO: Handle writes when Channel 3 is disabled | ||||
|     const base = if (!nr30.bank.read()) @as(u32, 0x10) else 0; // Write to the Opposite Bank in Use | ||||
|  | ||||
|     const i = base + addr - 0x0400_0090; | ||||
|     std.mem.writeIntSliceLittle(T, self.buf[i..][0..@sizeOf(T)], value); | ||||
| } | ||||
|  | ||||
| pub fn init(sched: *Scheduler) Self { | ||||
|     return .{ | ||||
|         .buf = [_]u8{0x00} ** buf_len, | ||||
|         .timer = 0, | ||||
|         .offset = 0, | ||||
|         .sched = sched, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn reset(self: *Self) void { | ||||
|     self.timer = 0; | ||||
|     self.offset = 0; | ||||
|  | ||||
|     // sample buffer isn't reset because it's outside of the range of what NR52{7}'s effects | ||||
| } | ||||
|  | ||||
| /// Reload internal Wave Timer | ||||
| pub fn reload(self: *Self, value: u11) void { | ||||
|     self.sched.removeScheduledEvent(.{ .ApuChannel = 2 }); | ||||
|  | ||||
|     self.timer = (@as(u16, 2048) - value) * 2; | ||||
|     self.sched.push(.{ .ApuChannel = 2 }, @as(u64, self.timer) * interval); | ||||
| } | ||||
|  | ||||
| /// Scheduler Event Handler | ||||
| pub fn onWaveTimerExpire(self: *Self, nrx34: io.Frequency, nr30: io.WaveSelect, late: u64) void { | ||||
|     if (nr30.dimension.read()) { | ||||
|         self.offset = (self.offset + 1) % 0x40; // 0x20 bytes (both banks), which contain 2 samples each | ||||
|     } else { | ||||
|         self.offset = (self.offset + 1) % 0x20; // 0x10 bytes, which contain 2 samples each | ||||
|     } | ||||
|  | ||||
|     self.timer = (@as(u16, 2048) - nrx34.frequency.read()) * 2; | ||||
|     self.sched.push(.{ .ApuChannel = 2 }, @as(u64, self.timer) * interval -| late); | ||||
| } | ||||
|  | ||||
| /// Generate Sample from Wave Synth | ||||
| pub fn sample(self: *const Self, nr30: io.WaveSelect) u4 { | ||||
|     const base = if (nr30.bank.read()) @as(u32, 0x10) else 0; | ||||
|  | ||||
|     const value = self.buf[base + self.offset / 2]; | ||||
|     return if (self.offset & 1 == 0) @truncate(u4, value >> 4) else @truncate(u4, value); | ||||
| } | ||||
|  | ||||
| /// TODO: Write comment | ||||
| pub fn shift(_: *const Self, nr32: io.WaveVolume) u2 { | ||||
|     return switch (nr32.kind.read()) { | ||||
|         0b00 => 3, // Mute / Zero | ||||
|         0b01 => 0, // 100% Volume | ||||
|         0b10 => 1, // 50% Volume | ||||
|         0b11 => 2, // 25% Volume | ||||
|     }; | ||||
| } | ||||
| @@ -1,63 +0,0 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
| const log = std.log.scoped(.Bios); | ||||
|  | ||||
| /// Size of the BIOS in bytes | ||||
| pub const size = 0x4000; | ||||
| const Self = @This(); | ||||
|  | ||||
| buf: ?[]u8, | ||||
| allocator: Allocator, | ||||
|  | ||||
| addr_latch: u32, | ||||
|  | ||||
| pub fn read(self: *Self, comptime T: type, r15: u32, addr: u32) T { | ||||
|     if (r15 < Self.size) { | ||||
|         self.addr_latch = addr; | ||||
|         return self._read(T, addr); | ||||
|     } | ||||
|  | ||||
|     log.debug("Rejected read since r15=0x{X:0>8}", .{r15}); | ||||
|     return @truncate(T, self._read(T, self.addr_latch)); | ||||
| } | ||||
|  | ||||
| pub fn dbgRead(self: *const Self, comptime T: type, r15: u32, addr: u32) T { | ||||
|     if (r15 < Self.size) return self._read(T, addr); | ||||
|     return @truncate(T, self._read(T, self.addr_latch + 8)); | ||||
| } | ||||
|  | ||||
| /// Read without the GBA safety checks | ||||
| fn _read(self: *const Self, comptime T: type, addr: u32) T { | ||||
|     const buf = self.buf orelse std.debug.panic("[BIOS] ZBA tried to read {} from 0x{X:0>8} but not BIOS was present", .{ T, addr }); | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32, u16, u8 => std.mem.readIntSliceLittle(T, buf[addr..][0..@sizeOf(T)]), | ||||
|         else => @compileError("BIOS: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(_: *Self, comptime T: type, addr: u32, value: T) void { | ||||
|     @setCold(true); | ||||
|     log.debug("Tried to write {} 0x{X:} to 0x{X:0>8} ", .{ T, value, addr }); | ||||
| } | ||||
|  | ||||
| pub fn init(allocator: Allocator, maybe_path: ?[]const u8) !Self { | ||||
|     const buf: ?[]u8 = if (maybe_path) |path| blk: { | ||||
|         const file = try std.fs.cwd().openFile(path, .{}); | ||||
|         defer file.close(); | ||||
|  | ||||
|         break :blk try file.readToEndAlloc(allocator, try file.getEndPos()); | ||||
|     } else null; | ||||
|  | ||||
|     return Self{ | ||||
|         .buf = buf, | ||||
|         .allocator = allocator, | ||||
|         .addr_latch = 0, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: *Self) void { | ||||
|     if (self.buf) |buf| self.allocator.free(buf); | ||||
|     self.* = undefined; | ||||
| } | ||||
| @@ -1,41 +0,0 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
| const ewram_size = 0x40000; | ||||
| const Self = @This(); | ||||
|  | ||||
| buf: []u8, | ||||
| allocator: Allocator, | ||||
|  | ||||
| pub fn read(self: *const Self, comptime T: type, address: usize) T { | ||||
|     const addr = address & 0x3FFFF; | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32, u16, u8 => std.mem.readIntSliceLittle(T, self.buf[addr..][0..@sizeOf(T)]), | ||||
|         else => @compileError("EWRAM: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(self: *const Self, comptime T: type, address: usize, value: T) void { | ||||
|     const addr = address & 0x3FFFF; | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32, u16, u8 => std.mem.writeIntSliceLittle(T, self.buf[addr..][0..@sizeOf(T)], value), | ||||
|         else => @compileError("EWRAM: Unsupported write width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn init(allocator: Allocator) !Self { | ||||
|     const buf = try allocator.alloc(u8, ewram_size); | ||||
|     std.mem.set(u8, buf, 0); | ||||
|  | ||||
|     return Self{ | ||||
|         .buf = buf, | ||||
|         .allocator = allocator, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: *Self) void { | ||||
|     self.allocator.free(self.buf); | ||||
|     self.* = undefined; | ||||
| } | ||||
| @@ -1,255 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const config = @import("../../config.zig"); | ||||
|  | ||||
| const Arm7tdmi = @import("../cpu.zig").Arm7tdmi; | ||||
| const Backup = @import("backup.zig").Backup; | ||||
| const Gpio = @import("gpio.zig").Gpio; | ||||
| const Allocator = std.mem.Allocator; | ||||
|  | ||||
| const log = std.log.scoped(.GamePak); | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| title: [12]u8, | ||||
| buf: []u8, | ||||
| allocator: Allocator, | ||||
| backup: Backup, | ||||
| gpio: *Gpio, | ||||
|  | ||||
| pub fn read(self: *Self, comptime T: type, address: u32) T { | ||||
|     const addr = address & 0x1FF_FFFF; | ||||
|  | ||||
|     if (self.backup.kind == .Eeprom) { | ||||
|         if (self.buf.len > 0x100_0000) { // Large | ||||
|             // Addresses 0x1FF_FF00 to 0x1FF_FFFF are reserved from EEPROM accesses if | ||||
|             // * Backup type is EEPROM | ||||
|             // * Large ROM (Size is greater than 16MB) | ||||
|             if (addr > 0x1FF_FEFF) | ||||
|                 return self.backup.eeprom.read(); | ||||
|         } else { | ||||
|             // Addresses 0x0D00_0000 to 0x0DFF_FFFF are reserved for EEPROM accesses if | ||||
|             // * Backup type is EEPROM | ||||
|             // * Small ROM (less than 16MB) | ||||
|             if (@truncate(u8, address >> 24) == 0x0D) | ||||
|                 return self.backup.eeprom.read(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (self.gpio.cnt == 1) { | ||||
|         // GPIO Can be read from | ||||
|         // We assume that this will only be true when a ROM actually does want something from GPIO | ||||
|  | ||||
|         switch (T) { | ||||
|             u32 => switch (address) { | ||||
|                 // TODO: Do I even need to implement these? | ||||
|                 0x0800_00C4 => std.debug.panic("Handle 32-bit GPIO Data/Direction Reads", .{}), | ||||
|                 0x0800_00C6 => std.debug.panic("Handle 32-bit GPIO Direction/Control Reads", .{}), | ||||
|                 0x0800_00C8 => std.debug.panic("Handle 32-bit GPIO Control Reads", .{}), | ||||
|                 else => {}, | ||||
|             }, | ||||
|             u16 => switch (address) { | ||||
|                 // FIXME: What do 16-bit GPIO Reads look like? | ||||
|                 0x0800_00C4 => return self.gpio.read(.Data), | ||||
|                 0x0800_00C6 => return self.gpio.read(.Direction), | ||||
|                 0x0800_00C8 => return self.gpio.read(.Control), | ||||
|                 else => {}, | ||||
|             }, | ||||
|             u8 => switch (address) { | ||||
|                 0x0800_00C4 => return self.gpio.read(.Data), | ||||
|                 0x0800_00C6 => return self.gpio.read(.Direction), | ||||
|                 0x0800_00C8 => return self.gpio.read(.Control), | ||||
|                 else => {}, | ||||
|             }, | ||||
|             else => @compileError("GamePak[GPIO]: Unsupported read width"), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => (@as(T, self.get(addr + 3)) << 24) | (@as(T, self.get(addr + 2)) << 16) | (@as(T, self.get(addr + 1)) << 8) | (@as(T, self.get(addr))), | ||||
|         u16 => (@as(T, self.get(addr + 1)) << 8) | @as(T, self.get(addr)), | ||||
|         u8 => self.get(addr), | ||||
|         else => @compileError("GamePak: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| inline fn get(self: *const Self, i: u32) u8 { | ||||
|     @setRuntimeSafety(false); | ||||
|     if (i < self.buf.len) return self.buf[i]; | ||||
|  | ||||
|     const lhs = i >> 1 & 0xFFFF; | ||||
|     return @truncate(u8, lhs >> 8 * @truncate(u5, i & 1)); | ||||
| } | ||||
|  | ||||
| pub fn dbgRead(self: *const Self, comptime T: type, address: u32) T { | ||||
|     const addr = address & 0x1FF_FFFF; | ||||
|  | ||||
|     if (self.backup.kind == .Eeprom) { | ||||
|         if (self.buf.len > 0x100_0000) { // Large | ||||
|             // Addresses 0x1FF_FF00 to 0x1FF_FFFF are reserved from EEPROM accesses if | ||||
|             // * Backup type is EEPROM | ||||
|             // * Large ROM (Size is greater than 16MB) | ||||
|             if (addr > 0x1FF_FEFF) | ||||
|                 return self.backup.eeprom.dbgRead(); | ||||
|         } else { | ||||
|             // Addresses 0x0D00_0000 to 0x0DFF_FFFF are reserved for EEPROM accesses if | ||||
|             // * Backup type is EEPROM | ||||
|             // * Small ROM (less than 16MB) | ||||
|             if (@truncate(u8, address >> 24) == 0x0D) | ||||
|                 return self.backup.eeprom.dbgRead(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (self.gpio.cnt == 1) { | ||||
|         // GPIO Can be read from | ||||
|         // We assume that this will only be true when a ROM actually does want something from GPIO | ||||
|  | ||||
|         switch (T) { | ||||
|             u32 => switch (address) { | ||||
|                 // FIXME: Do I even need to implement these? | ||||
|                 0x0800_00C4 => std.debug.panic("Handle 32-bit GPIO Data/Direction Reads", .{}), | ||||
|                 0x0800_00C6 => std.debug.panic("Handle 32-bit GPIO Direction/Control Reads", .{}), | ||||
|                 0x0800_00C8 => std.debug.panic("Handle 32-bit GPIO Control Reads", .{}), | ||||
|                 else => {}, | ||||
|             }, | ||||
|             u16 => switch (address) { | ||||
|                 0x0800_00C4 => return self.gpio.read(.Data), | ||||
|                 0x0800_00C6 => return self.gpio.read(.Direction), | ||||
|                 0x0800_00C8 => return self.gpio.read(.Control), | ||||
|                 else => {}, | ||||
|             }, | ||||
|             u8 => switch (address) { | ||||
|                 0x0800_00C4 => return self.gpio.read(.Data), | ||||
|                 0x0800_00C6 => return self.gpio.read(.Direction), | ||||
|                 0x0800_00C8 => return self.gpio.read(.Control), | ||||
|                 else => {}, | ||||
|             }, | ||||
|             else => @compileError("GamePak[GPIO]: Unsupported read width"), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => (@as(T, self.get(addr + 3)) << 24) | (@as(T, self.get(addr + 2)) << 16) | (@as(T, self.get(addr + 1)) << 8) | (@as(T, self.get(addr))), | ||||
|         u16 => (@as(T, self.get(addr + 1)) << 8) | @as(T, self.get(addr)), | ||||
|         u8 => self.get(addr), | ||||
|         else => @compileError("GamePak: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(self: *Self, comptime T: type, word_count: u16, address: u32, value: T) void { | ||||
|     const addr = address & 0x1FF_FFFF; | ||||
|  | ||||
|     if (self.backup.kind == .Eeprom) { | ||||
|         const bit = @truncate(u1, value); | ||||
|  | ||||
|         if (self.buf.len > 0x100_0000) { // Large | ||||
|             // Addresses 0x1FF_FF00 to 0x1FF_FFFF are reserved from EEPROM accesses if | ||||
|             // * Backup type is EEPROM | ||||
|             // * Large ROM (Size is greater than 16MB) | ||||
|             if (addr > 0x1FF_FEFF) | ||||
|                 return self.backup.eeprom.write(word_count, &self.backup.buf, bit); | ||||
|         } else { | ||||
|             // Addresses 0x0D00_0000 to 0x0DFF_FFFF are reserved for EEPROM accesses if | ||||
|             // * Backup type is EEPROM | ||||
|             // * Small ROM (less than 16MB) | ||||
|             if (@truncate(u8, address >> 24) == 0x0D) | ||||
|                 return self.backup.eeprom.write(word_count, &self.backup.buf, bit); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     switch (T) { | ||||
|         u32 => switch (address) { | ||||
|             0x0800_00C4 => { | ||||
|                 self.gpio.write(.Data, @truncate(u4, value)); | ||||
|                 self.gpio.write(.Direction, @truncate(u4, value >> 16)); | ||||
|             }, | ||||
|             0x0800_00C6 => { | ||||
|                 self.gpio.write(.Direction, @truncate(u4, value)); | ||||
|                 self.gpio.write(.Control, @truncate(u1, value >> 16)); | ||||
|             }, | ||||
|             else => log.err("Wrote {} 0x{X:0>8} to 0x{X:0>8}, Unhandled", .{ T, value, address }), | ||||
|         }, | ||||
|         u16 => switch (address) { | ||||
|             0x0800_00C4 => self.gpio.write(.Data, @truncate(u4, value)), | ||||
|             0x0800_00C6 => self.gpio.write(.Direction, @truncate(u4, value)), | ||||
|             0x0800_00C8 => self.gpio.write(.Control, @truncate(u1, value)), | ||||
|             else => log.err("Wrote {} 0x{X:0>4} to 0x{X:0>8}, Unhandled", .{ T, value, address }), | ||||
|         }, | ||||
|         u8 => log.debug("Wrote {} 0x{X:0>2} to 0x{X:0>8}, Ignored.", .{ T, value, address }), | ||||
|         else => @compileError("GamePak: Unsupported write width"), | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn init(allocator: Allocator, cpu: *Arm7tdmi, rom_path: []const u8, save_path: ?[]const u8) !Self { | ||||
|     const file = try std.fs.cwd().openFile(rom_path, .{}); | ||||
|     defer file.close(); | ||||
|  | ||||
|     const file_buf = try file.readToEndAlloc(allocator, try file.getEndPos()); | ||||
|     const title = file_buf[0xA0..0xAC].*; | ||||
|     const kind = Backup.guess(file_buf); | ||||
|     const device = if (config.config().guest.force_rtc) .Rtc else guessDevice(file_buf); | ||||
|  | ||||
|     logHeader(file_buf, &title); | ||||
|  | ||||
|     return .{ | ||||
|         .buf = file_buf, | ||||
|         .allocator = allocator, | ||||
|         .title = title, | ||||
|         .backup = try Backup.init(allocator, kind, title, save_path), | ||||
|         .gpio = try Gpio.init(allocator, cpu, device), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: *Self) void { | ||||
|     self.backup.deinit(); | ||||
|     self.gpio.deinit(self.allocator); | ||||
|     self.allocator.destroy(self.gpio); | ||||
|     self.allocator.free(self.buf); | ||||
|     self.* = undefined; | ||||
| } | ||||
|  | ||||
| /// Searches the ROM to see if it can determine whether the ROM it's searching uses | ||||
| /// any GPIO device, like a RTC for example. | ||||
| fn guessDevice(buf: []const u8) Gpio.Device.Kind { | ||||
|     // Try to Guess if ROM uses RTC | ||||
|     const needle = "RTC_V"; // I was told SIIRTC_V, though Pokemen Firered (USA) is a false negative | ||||
|  | ||||
|     var i: usize = 0; | ||||
|     while ((i + needle.len) < buf.len) : (i += 1) { | ||||
|         if (std.mem.eql(u8, needle, buf[i..(i + needle.len)])) return .Rtc; | ||||
|     } | ||||
|  | ||||
|     // TODO: Detect other GPIO devices | ||||
|  | ||||
|     return .None; | ||||
| } | ||||
|  | ||||
| fn logHeader(buf: []const u8, title: *const [12]u8) void { | ||||
|     const code = buf[0xAC..0xB0]; | ||||
|     const maker = buf[0xB0..0xB2]; | ||||
|     const version = buf[0xBC]; | ||||
|  | ||||
|     log.info("Title: {s}", .{title}); | ||||
|     if (version != 0) log.info("Version: {}", .{version}); | ||||
|     log.info("Game Code: {s}", .{code}); | ||||
|     log.info("Maker Code: {s}", .{maker}); | ||||
| } | ||||
|  | ||||
| test "OOB Access" { | ||||
|     const title = .{ 'H', 'E', 'L', 'L', 'O', ' ', 'W', 'O', 'R', 'L', 'D', '!' }; | ||||
|     const alloc = std.testing.allocator; | ||||
|     const pak = Self{ | ||||
|         .buf = &.{}, | ||||
|         .alloc = alloc, | ||||
|         .title = title, | ||||
|         .backup = try Backup.init(alloc, .None, title, null), | ||||
|     }; | ||||
|  | ||||
|     std.debug.assert(pak.get(0) == 0x00); // 0x0000 | ||||
|     std.debug.assert(pak.get(1) == 0x00); | ||||
|  | ||||
|     std.debug.assert(pak.get(2) == 0x01); // 0x0001 | ||||
|     std.debug.assert(pak.get(3) == 0x00); | ||||
|  | ||||
|     std.debug.assert(pak.get(4) == 0x02); // 0x0002 | ||||
|     std.debug.assert(pak.get(5) == 0x00); | ||||
| } | ||||
| @@ -1,41 +0,0 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
| const iwram_size = 0x8000; | ||||
| const Self = @This(); | ||||
|  | ||||
| buf: []u8, | ||||
| allocator: Allocator, | ||||
|  | ||||
| pub fn read(self: *const Self, comptime T: type, address: usize) T { | ||||
|     const addr = address & 0x7FFF; | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32, u16, u8 => std.mem.readIntSliceLittle(T, self.buf[addr..][0..@sizeOf(T)]), | ||||
|         else => @compileError("IWRAM: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(self: *const Self, comptime T: type, address: usize, value: T) void { | ||||
|     const addr = address & 0x7FFF; | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32, u16, u8 => std.mem.writeIntSliceLittle(T, self.buf[addr..][0..@sizeOf(T)], value), | ||||
|         else => @compileError("IWRAM: Unsupported write width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn init(allocator: Allocator) !Self { | ||||
|     const buf = try allocator.alloc(u8, iwram_size); | ||||
|     std.mem.set(u8, buf, 0); | ||||
|  | ||||
|     return Self{ | ||||
|         .buf = buf, | ||||
|         .allocator = allocator, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn deinit(self: *Self) void { | ||||
|     self.allocator.free(self.buf); | ||||
|     self.* = undefined; | ||||
| } | ||||
| @@ -1,218 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const Allocator = std.mem.Allocator; | ||||
| const log = std.log.scoped(.Backup); | ||||
|  | ||||
| const Eeprom = @import("backup/eeprom.zig").Eeprom; | ||||
| const Flash = @import("backup/Flash.zig"); | ||||
|  | ||||
| const escape = @import("../../util.zig").escape; | ||||
|  | ||||
| const Needle = struct { str: []const u8, kind: Backup.Kind }; | ||||
| const backup_kinds = [6]Needle{ | ||||
|     .{ .str = "EEPROM_V", .kind = .Eeprom }, | ||||
|     .{ .str = "SRAM_V", .kind = .Sram }, | ||||
|     .{ .str = "SRAM_F_V", .kind = .Sram }, | ||||
|     .{ .str = "FLASH_V", .kind = .Flash }, | ||||
|     .{ .str = "FLASH512_V", .kind = .Flash }, | ||||
|     .{ .str = "FLASH1M_V", .kind = .Flash1M }, | ||||
| }; | ||||
|  | ||||
| const SaveError = error{Unsupported}; | ||||
|  | ||||
| pub const Backup = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     buf: []u8, | ||||
|     allocator: Allocator, | ||||
|     kind: Kind, | ||||
|  | ||||
|     title: [12]u8, | ||||
|     save_path: ?[]const u8, | ||||
|  | ||||
|     flash: Flash, | ||||
|     eeprom: Eeprom, | ||||
|  | ||||
|     const Kind = enum { | ||||
|         Eeprom, | ||||
|         Sram, | ||||
|         Flash, | ||||
|         Flash1M, | ||||
|         None, | ||||
|     }; | ||||
|  | ||||
|     pub fn read(self: *const Self, address: usize) u8 { | ||||
|         const addr = address & 0xFFFF; | ||||
|  | ||||
|         switch (self.kind) { | ||||
|             .Flash => { | ||||
|                 switch (addr) { | ||||
|                     0x0000 => if (self.flash.id_mode) return 0x32, // Panasonic manufacturer ID | ||||
|                     0x0001 => if (self.flash.id_mode) return 0x1B, // Panasonic device ID | ||||
|                     else => {}, | ||||
|                 } | ||||
|  | ||||
|                 return self.flash.read(self.buf, addr); | ||||
|             }, | ||||
|             .Flash1M => { | ||||
|                 switch (addr) { | ||||
|                     0x0000 => if (self.flash.id_mode) return 0x62, // Sanyo manufacturer ID | ||||
|                     0x0001 => if (self.flash.id_mode) return 0x13, // Sanyo device ID | ||||
|                     else => {}, | ||||
|                 } | ||||
|  | ||||
|                 return self.flash.read(self.buf, addr); | ||||
|             }, | ||||
|             .Sram => return self.buf[addr & 0x7FFF], // 32K SRAM chip is mirrored | ||||
|             .None, .Eeprom => return 0xFF, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn write(self: *Self, address: usize, byte: u8) void { | ||||
|         const addr = address & 0xFFFF; | ||||
|  | ||||
|         switch (self.kind) { | ||||
|             .Flash, .Flash1M => { | ||||
|                 if (self.flash.prep_write) return self.flash.write(self.buf, addr, byte); | ||||
|                 if (self.flash.shouldEraseSector(addr, byte)) return self.flash.erase(self.buf, addr); | ||||
|  | ||||
|                 switch (addr) { | ||||
|                     0x0000 => if (self.kind == .Flash1M and self.flash.set_bank) { | ||||
|                         self.flash.bank = @truncate(u1, byte); | ||||
|                     }, | ||||
|                     0x5555 => { | ||||
|                         if (self.flash.state == .Command) { | ||||
|                             self.flash.handleCommand(self.buf, byte); | ||||
|                         } else if (byte == 0xAA and self.flash.state == .Ready) { | ||||
|                             self.flash.state = .Set; | ||||
|                         } else if (byte == 0xF0) { | ||||
|                             self.flash.state = .Ready; | ||||
|                         } | ||||
|                     }, | ||||
|                     0x2AAA => if (byte == 0x55 and self.flash.state == .Set) { | ||||
|                         self.flash.state = .Command; | ||||
|                     }, | ||||
|                     else => {}, | ||||
|                 } | ||||
|             }, | ||||
|             .Sram => self.buf[addr & 0x7FFF] = byte, | ||||
|             .None, .Eeprom => {}, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn init(allocator: Allocator, kind: Kind, title: [12]u8, path: ?[]const u8) !Self { | ||||
|         log.info("Kind: {}", .{kind}); | ||||
|  | ||||
|         const buf_size: usize = switch (kind) { | ||||
|             .Sram => 0x8000, // 32K | ||||
|             .Flash => 0x10000, // 64K | ||||
|             .Flash1M => 0x20000, // 128K | ||||
|             .None, .Eeprom => 0, // EEPROM is handled upon first Read Request to it | ||||
|         }; | ||||
|  | ||||
|         const buf = try allocator.alloc(u8, buf_size); | ||||
|         std.mem.set(u8, buf, 0xFF); | ||||
|  | ||||
|         var backup = Self{ | ||||
|             .buf = buf, | ||||
|             .allocator = allocator, | ||||
|             .kind = kind, | ||||
|             .title = title, | ||||
|             .save_path = path, | ||||
|             .flash = Flash.create(), | ||||
|             .eeprom = Eeprom.create(allocator), | ||||
|         }; | ||||
|  | ||||
|         if (backup.save_path) |p| backup.readSave(allocator, p) catch |e| log.err("Failed to load save: {}", .{e}); | ||||
|         return backup; | ||||
|     } | ||||
|  | ||||
|     pub fn deinit(self: *Self) void { | ||||
|         if (self.save_path) |path| self.writeSave(self.allocator, path) catch |e| log.err("Failed to write save: {}", .{e}); | ||||
|         self.allocator.free(self.buf); | ||||
|         self.* = undefined; | ||||
|     } | ||||
|  | ||||
|     /// Guesses the Backup Kind of a GBA ROM | ||||
|     pub fn guess(rom: []const u8) Kind { | ||||
|         for (backup_kinds) |needle| { | ||||
|             const needle_len = needle.str.len; | ||||
|  | ||||
|             var i: usize = 0; | ||||
|             while ((i + needle_len) < rom.len) : (i += 1) { | ||||
|                 if (std.mem.eql(u8, needle.str, rom[i..][0..needle_len])) return needle.kind; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return .None; | ||||
|     } | ||||
|  | ||||
|     fn readSave(self: *Self, allocator: Allocator, path: []const u8) !void { | ||||
|         const file_path = try self.savePath(allocator, path); | ||||
|         defer allocator.free(file_path); | ||||
|  | ||||
|         const expected = "untitled.sav"; | ||||
|         if (std.mem.eql(u8, file_path[file_path.len - expected.len .. file_path.len], expected)) { | ||||
|             return log.err("ROM header lacks title, no save loaded", .{}); | ||||
|         } | ||||
|  | ||||
|         const file: std.fs.File = try std.fs.openFileAbsolute(file_path, .{}); | ||||
|         const file_buf = try file.readToEndAlloc(allocator, try file.getEndPos()); | ||||
|         defer allocator.free(file_buf); | ||||
|  | ||||
|         switch (self.kind) { | ||||
|             .Sram, .Flash, .Flash1M => { | ||||
|                 if (self.buf.len == file_buf.len) { | ||||
|                     std.mem.copy(u8, self.buf, file_buf); | ||||
|                     return log.info("Loaded Save from {s}", .{file_path}); | ||||
|                 } | ||||
|  | ||||
|                 log.err("{s} is {} bytes, but we expected {} bytes", .{ file_path, file_buf.len, self.buf.len }); | ||||
|             }, | ||||
|             .Eeprom => { | ||||
|                 if (file_buf.len == 0x200 or file_buf.len == 0x2000) { | ||||
|                     self.eeprom.kind = if (file_buf.len == 0x200) .Small else .Large; | ||||
|  | ||||
|                     self.buf = try allocator.alloc(u8, file_buf.len); | ||||
|                     std.mem.copy(u8, self.buf, file_buf); | ||||
|                     return log.info("Loaded Save from {s}", .{file_path}); | ||||
|                 } | ||||
|  | ||||
|                 log.err("EEPROM can either be 0x200 bytes or 0x2000 byes, but {s} was {X:} bytes", .{ | ||||
|                     file_path, | ||||
|                     file_buf.len, | ||||
|                 }); | ||||
|             }, | ||||
|             .None => return SaveError.Unsupported, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn savePath(self: *const Self, allocator: Allocator, path: []const u8) ![]const u8 { | ||||
|         const filename = try self.saveName(allocator); | ||||
|         defer allocator.free(filename); | ||||
|  | ||||
|         return try std.fs.path.join(allocator, &[_][]const u8{ path, filename }); | ||||
|     } | ||||
|  | ||||
|     fn saveName(self: *const Self, allocator: Allocator) ![]const u8 { | ||||
|         const title_str = std.mem.sliceTo(&escape(self.title), 0); | ||||
|         const name = if (title_str.len != 0) title_str else "untitled"; | ||||
|  | ||||
|         return try std.mem.concat(allocator, u8, &[_][]const u8{ name, ".sav" }); | ||||
|     } | ||||
|  | ||||
|     fn writeSave(self: Self, allocator: Allocator, path: []const u8) !void { | ||||
|         const file_path = try self.savePath(allocator, path); | ||||
|         defer allocator.free(file_path); | ||||
|  | ||||
|         switch (self.kind) { | ||||
|             .Sram, .Flash, .Flash1M, .Eeprom => { | ||||
|                 const file = try std.fs.createFileAbsolute(file_path, .{}); | ||||
|                 defer file.close(); | ||||
|  | ||||
|                 try file.writeAll(self.buf); | ||||
|                 log.info("Wrote Save to {s}", .{file_path}); | ||||
|             }, | ||||
|             else => return SaveError.Unsupported, | ||||
|         } | ||||
|     } | ||||
| }; | ||||
| @@ -1,72 +0,0 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Self = @This(); | ||||
|  | ||||
| state: State, | ||||
|  | ||||
| id_mode: bool, | ||||
| set_bank: bool, | ||||
| prep_erase: bool, | ||||
| prep_write: bool, | ||||
|  | ||||
| bank: u1, | ||||
|  | ||||
| const State = enum { | ||||
|     Ready, | ||||
|     Set, | ||||
|     Command, | ||||
| }; | ||||
|  | ||||
| pub fn read(self: *const Self, buf: []u8, idx: usize) u8 { | ||||
|     return buf[self.address() + idx]; | ||||
| } | ||||
|  | ||||
| pub fn write(self: *Self, buf: []u8, idx: usize, byte: u8) void { | ||||
|     buf[self.address() + idx] = byte; | ||||
|     self.prep_write = false; | ||||
| } | ||||
|  | ||||
| pub fn create() Self { | ||||
|     return .{ | ||||
|         .state = .Ready, | ||||
|         .id_mode = false, | ||||
|         .set_bank = false, | ||||
|         .prep_erase = false, | ||||
|         .prep_write = false, | ||||
|         .bank = 0, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn handleCommand(self: *Self, buf: []u8, byte: u8) void { | ||||
|     switch (byte) { | ||||
|         0x90 => self.id_mode = true, | ||||
|         0xF0 => self.id_mode = false, | ||||
|         0xB0 => self.set_bank = true, | ||||
|         0x80 => self.prep_erase = true, | ||||
|         0x10 => { | ||||
|             std.mem.set(u8, buf, 0xFF); | ||||
|             self.prep_erase = false; | ||||
|         }, | ||||
|         0xA0 => self.prep_write = true, | ||||
|         else => std.debug.panic("Unhandled Flash Command: 0x{X:0>2}", .{byte}), | ||||
|     } | ||||
|  | ||||
|     self.state = .Ready; | ||||
| } | ||||
|  | ||||
| pub fn shouldEraseSector(self: *const Self, addr: usize, byte: u8) bool { | ||||
|     return self.state == .Command and self.prep_erase and byte == 0x30 and addr & 0xFFF == 0x000; | ||||
| } | ||||
|  | ||||
| pub fn erase(self: *Self, buf: []u8, sector: usize) void { | ||||
|     const start = self.address() + (sector & 0xF000); | ||||
|  | ||||
|     std.mem.set(u8, buf[start..][0..0x1000], 0xFF); | ||||
|     self.prep_erase = false; | ||||
|     self.state = .Ready; | ||||
| } | ||||
|  | ||||
| /// Base Address | ||||
| inline fn address(self: *const Self) usize { | ||||
|     return if (self.bank == 1) 0x10000 else @as(usize, 0); | ||||
| } | ||||
| @@ -1,269 +0,0 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
|  | ||||
| const log = std.log.scoped(.Eeprom); | ||||
|  | ||||
| pub const Eeprom = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     addr: u14, | ||||
|  | ||||
|     kind: Kind, | ||||
|     state: State, | ||||
|     writer: Writer, | ||||
|     reader: Reader, | ||||
|  | ||||
|     allocator: Allocator, | ||||
|  | ||||
|     const Kind = enum { | ||||
|         Unknown, | ||||
|         Small, // 512B | ||||
|         Large, // 8KB | ||||
|     }; | ||||
|  | ||||
|     const State = enum { | ||||
|         Ready, | ||||
|         Read, | ||||
|         Write, | ||||
|         WriteTransfer, | ||||
|         RequestEnd, | ||||
|     }; | ||||
|  | ||||
|     pub fn read(self: *Self) u1 { | ||||
|         return self.reader.read(); | ||||
|     } | ||||
|  | ||||
|     pub fn dbgRead(self: *const Self) u1 { | ||||
|         return self.reader.dbgRead(); | ||||
|     } | ||||
|  | ||||
|     pub fn write(self: *Self, word_count: u16, buf: *[]u8, bit: u1) void { | ||||
|         if (self.guessKind(word_count)) |found| { | ||||
|             log.info("EEPROM Kind: {}", .{found}); | ||||
|             self.kind = found; | ||||
|  | ||||
|             // buf.len will not equal zero when a save file was found and loaded. | ||||
|             // Right now, we assume that the save file is of the correct size which | ||||
|             // isn't necessarily true, since we can't trust anything a user can influence | ||||
|             // TODO: use ?[]u8 instead of a 0-sized slice? | ||||
|             if (buf.len == 0) { | ||||
|                 const len: usize = switch (found) { | ||||
|                     .Small => 0x200, | ||||
|                     .Large => 0x2000, | ||||
|                     else => unreachable, | ||||
|                 }; | ||||
|  | ||||
|                 buf.* = self.allocator.alloc(u8, len) catch |e| { | ||||
|                     log.err("Failed to resize EEPROM buf to {} bytes", .{len}); | ||||
|                     std.debug.panic("EEPROM entered irrecoverable state {}", .{e}); | ||||
|                 }; | ||||
|                 std.mem.set(u8, buf.*, 0xFF); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (self.state == .RequestEnd) { | ||||
|             // if (bit != 0) log.debug("EEPROM Request did not end in 0u1. TODO: is this ok?", .{}); | ||||
|             self.state = .Ready; | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         switch (self.state) { | ||||
|             .Ready => self.writer.requestWrite(bit), | ||||
|             .Read, .Write => self.writer.addressWrite(self.kind, bit), | ||||
|             .WriteTransfer => self.writer.dataWrite(bit), | ||||
|             .RequestEnd => unreachable, // We return early just above this block | ||||
|         } | ||||
|  | ||||
|         self.tick(buf.*); | ||||
|     } | ||||
|  | ||||
|     pub fn create(allocator: Allocator) Self { | ||||
|         return .{ | ||||
|             .kind = .Unknown, | ||||
|             .state = .Ready, | ||||
|             .writer = Writer.create(), | ||||
|             .reader = Reader.create(), | ||||
|             .addr = 0, | ||||
|             .allocator = allocator, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn guessKind(self: *const Self, word_count: u16) ?Kind { | ||||
|         if (self.kind != .Unknown or self.state != .Read) return null; | ||||
|  | ||||
|         return switch (word_count) { | ||||
|             17 => .Large, | ||||
|             9 => .Small, | ||||
|             else => blk: { | ||||
|                 log.err("Unexpected length of DMA3 Transfer upon initial EEPROM read: {}", .{word_count}); | ||||
|                 break :blk null; | ||||
|             }, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn tick(self: *Self, buf: []u8) void { | ||||
|         switch (self.state) { | ||||
|             .Ready => { | ||||
|                 if (self.writer.len() == 2) { | ||||
|                     const req = @intCast(u2, self.writer.finish()); | ||||
|                     switch (req) { | ||||
|                         0b11 => self.state = .Read, | ||||
|                         0b10 => self.state = .Write, | ||||
|                         else => log.err("Unknown EEPROM Request 0b{b:0>2}", .{req}), | ||||
|                     } | ||||
|                 } | ||||
|             }, | ||||
|             .Read => { | ||||
|                 switch (self.kind) { | ||||
|                     .Large => { | ||||
|                         if (self.writer.len() == 14) { | ||||
|                             const addr = @intCast(u10, self.writer.finish()); | ||||
|                             const value = std.mem.readIntSliceLittle(u64, buf[@as(u13, addr) * 8 ..][0..8]); | ||||
|  | ||||
|                             self.reader.configure(value); | ||||
|                             self.state = .RequestEnd; | ||||
|                         } | ||||
|                     }, | ||||
|                     .Small => { | ||||
|                         if (self.writer.len() == 6) { | ||||
|                             // FIXME: Duplicated code from above | ||||
|                             const addr = @intCast(u6, self.writer.finish()); | ||||
|                             const value = std.mem.readIntSliceLittle(u64, buf[@as(u13, addr) * 8 ..][0..8]); | ||||
|  | ||||
|                             self.reader.configure(value); | ||||
|                             self.state = .RequestEnd; | ||||
|                         } | ||||
|                     }, | ||||
|                     else => log.err("Unable to calculate EEPROM read address. EEPROM size UNKNOWN", .{}), | ||||
|                 } | ||||
|             }, | ||||
|             .Write => { | ||||
|                 switch (self.kind) { | ||||
|                     .Large => { | ||||
|                         if (self.writer.len() == 14) { | ||||
|                             self.addr = @intCast(u10, self.writer.finish()); | ||||
|                             self.state = .WriteTransfer; | ||||
|                         } | ||||
|                     }, | ||||
|                     .Small => { | ||||
|                         if (self.writer.len() == 6) { | ||||
|                             self.addr = @intCast(u6, self.writer.finish()); | ||||
|                             self.state = .WriteTransfer; | ||||
|                         } | ||||
|                     }, | ||||
|                     else => log.err("Unable to calculate EEPROM write address. EEPROM size UNKNOWN", .{}), | ||||
|                 } | ||||
|             }, | ||||
|             .WriteTransfer => { | ||||
|                 if (self.writer.len() == 64) { | ||||
|                     std.mem.writeIntSliceLittle(u64, buf[self.addr * 8 ..][0..8], self.writer.finish()); | ||||
|                     self.state = .RequestEnd; | ||||
|                 } | ||||
|             }, | ||||
|             .RequestEnd => unreachable, // We return early in write() if state is .RequestEnd | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Reader = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     data: u64, | ||||
|     i: u8, | ||||
|     enabled: bool, | ||||
|  | ||||
|     fn create() Self { | ||||
|         return .{ | ||||
|             .data = 0, | ||||
|             .i = 0, | ||||
|             .enabled = false, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn read(self: *Self) u1 { | ||||
|         if (!self.enabled) return 1; | ||||
|  | ||||
|         const bit = if (self.i < 4) blk: { | ||||
|             break :blk 0; | ||||
|         } else blk: { | ||||
|             const idx = @intCast(u6, 63 - (self.i - 4)); | ||||
|             break :blk @truncate(u1, self.data >> idx); | ||||
|         }; | ||||
|  | ||||
|         self.i = (self.i + 1) % (64 + 4); | ||||
|         if (self.i == 0) self.enabled = false; | ||||
|  | ||||
|         return bit; | ||||
|     } | ||||
|  | ||||
|     fn dbgRead(self: *const Self) u1 { | ||||
|         if (!self.enabled) return 1; | ||||
|  | ||||
|         const bit = if (self.i < 4) blk: { | ||||
|             break :blk 0; | ||||
|         } else blk: { | ||||
|             const idx = @intCast(u6, 63 - (self.i - 4)); | ||||
|             break :blk @truncate(u1, self.data >> idx); | ||||
|         }; | ||||
|  | ||||
|         return bit; | ||||
|     } | ||||
|  | ||||
|     fn configure(self: *Self, value: u64) void { | ||||
|         self.data = value; | ||||
|         self.i = 0; | ||||
|         self.enabled = true; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Writer = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     data: u64, | ||||
|     i: u8, | ||||
|  | ||||
|     fn create() Self { | ||||
|         return .{ .data = 0, .i = 0 }; | ||||
|     } | ||||
|  | ||||
|     fn requestWrite(self: *Self, bit: u1) void { | ||||
|         const idx = @intCast(u1, 1 - self.i); | ||||
|         self.data = (self.data & ~(@as(u64, 1) << idx)) | (@as(u64, bit) << idx); | ||||
|         self.i += 1; | ||||
|     } | ||||
|  | ||||
|     fn addressWrite(self: *Self, kind: Eeprom.Kind, bit: u1) void { | ||||
|         if (kind == .Unknown) return; | ||||
|  | ||||
|         const size: u4 = switch (kind) { | ||||
|             .Large => 13, | ||||
|             .Small => 5, | ||||
|             .Unknown => unreachable, | ||||
|         }; | ||||
|  | ||||
|         const idx = @intCast(u4, size - self.i); | ||||
|         self.data = (self.data & ~(@as(u64, 1) << idx)) | (@as(u64, bit) << idx); | ||||
|         self.i += 1; | ||||
|     } | ||||
|  | ||||
|     fn dataWrite(self: *Self, bit: u1) void { | ||||
|         const idx = @intCast(u6, 63 - self.i); | ||||
|         self.data = (self.data & ~(@as(u64, 1) << idx)) | (@as(u64, bit) << idx); | ||||
|         self.i += 1; | ||||
|     } | ||||
|  | ||||
|     fn len(self: *const Self) u8 { | ||||
|         return self.i; | ||||
|     } | ||||
|  | ||||
|     fn finish(self: *Self) u64 { | ||||
|         defer self.reset(); | ||||
|         return self.data; | ||||
|     } | ||||
|  | ||||
|     fn reset(self: *Self) void { | ||||
|         self.i = 0; | ||||
|         self.data = 0; | ||||
|     } | ||||
| }; | ||||
| @@ -1,360 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const util = @import("../../util.zig"); | ||||
|  | ||||
| const DmaControl = @import("io.zig").DmaControl; | ||||
| const Bus = @import("../Bus.zig"); | ||||
| const Arm7tdmi = @import("../cpu.zig").Arm7tdmi; | ||||
|  | ||||
| pub const DmaTuple = struct { DmaController(0), DmaController(1), DmaController(2), DmaController(3) }; | ||||
| const log = std.log.scoped(.DmaTransfer); | ||||
|  | ||||
| const getHalf = util.getHalf; | ||||
| const setHalf = util.setHalf; | ||||
| const setQuart = util.setQuart; | ||||
|  | ||||
| const rotr = @import("../../util.zig").rotr; | ||||
|  | ||||
| pub fn create() DmaTuple { | ||||
|     return .{ DmaController(0).init(), DmaController(1).init(), DmaController(2).init(), DmaController(3).init() }; | ||||
| } | ||||
|  | ||||
| pub fn read(comptime T: type, dma: *const DmaTuple, addr: u32) ?T { | ||||
|     const byte_addr = @truncate(u8, addr); | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => switch (byte_addr) { | ||||
|             0xB0, 0xB4 => null, // DMA0SAD, DMA0DAD, | ||||
|             0xB8 => @as(T, dma.*[0].dmacntH()) << 16, // DMA0CNT_L is write-only | ||||
|             0xBC, 0xC0 => null, // DMA1SAD, DMA1DAD | ||||
|             0xC4 => @as(T, dma.*[1].dmacntH()) << 16, // DMA1CNT_L is write-only | ||||
|             0xC8, 0xCC => null, // DMA2SAD, DMA2DAD | ||||
|             0xD0 => @as(T, dma.*[2].dmacntH()) << 16, // DMA2CNT_L is write-only | ||||
|             0xD4, 0xD8 => null, // DMA3SAD, DMA3DAD | ||||
|             0xDC => @as(T, dma.*[3].dmacntH()) << 16, // DMA3CNT_L is write-only | ||||
|             else => util.io.read.err(T, log, "unaligned {} read from 0x{X:0>8}", .{ T, addr }), | ||||
|         }, | ||||
|         u16 => switch (byte_addr) { | ||||
|             0xB0, 0xB2, 0xB4, 0xB6 => null, // DMA0SAD, DMA0DAD | ||||
|             0xB8 => 0x0000, // DMA0CNT_L, suite.gba expects 0x0000 instead of 0xDEAD | ||||
|             0xBA => dma.*[0].dmacntH(), | ||||
|  | ||||
|             0xBC, 0xBE, 0xC0, 0xC2 => null, // DMA1SAD, DMA1DAD | ||||
|             0xC4 => 0x0000, // DMA1CNT_L | ||||
|             0xC6 => dma.*[1].dmacntH(), | ||||
|  | ||||
|             0xC8, 0xCA, 0xCC, 0xCE => null, // DMA2SAD, DMA2DAD | ||||
|             0xD0 => 0x0000, // DMA2CNT_L | ||||
|             0xD2 => dma.*[2].dmacntH(), | ||||
|  | ||||
|             0xD4, 0xD6, 0xD8, 0xDA => null, // DMA3SAD, DMA3DAD | ||||
|             0xDC => 0x0000, // DMA3CNT_L | ||||
|             0xDE => dma.*[3].dmacntH(), | ||||
|             else => util.io.read.err(T, log, "unaligned {} read from 0x{X:0>8}", .{ T, addr }), | ||||
|         }, | ||||
|         u8 => switch (byte_addr) { | ||||
|             0xB0...0xB7 => null, // DMA0SAD, DMA0DAD | ||||
|             0xB8, 0xB9 => 0x00, // DMA0CNT_L | ||||
|             0xBA, 0xBB => @truncate(T, dma.*[0].dmacntH() >> getHalf(byte_addr)), | ||||
|  | ||||
|             0xBC...0xC3 => null, // DMA1SAD, DMA1DAD | ||||
|             0xC4, 0xC5 => 0x00, // DMA1CNT_L | ||||
|             0xC6, 0xC7 => @truncate(T, dma.*[1].dmacntH() >> getHalf(byte_addr)), | ||||
|  | ||||
|             0xC8...0xCF => null, // DMA2SAD, DMA2DAD | ||||
|             0xD0, 0xD1 => 0x00, // DMA2CNT_L | ||||
|             0xD2, 0xD3 => @truncate(T, dma.*[2].dmacntH() >> getHalf(byte_addr)), | ||||
|  | ||||
|             0xD4...0xDB => null, // DMA3SAD, DMA3DAD | ||||
|             0xDC, 0xDD => 0x00, // DMA3CNT_L | ||||
|             0xDE, 0xDF => @truncate(T, dma.*[3].dmacntH() >> getHalf(byte_addr)), | ||||
|             else => util.io.read.err(T, log, "unexpected {} read from 0x{X:0>8}", .{ T, addr }), | ||||
|         }, | ||||
|         else => @compileError("DMA: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(comptime T: type, dma: *DmaTuple, addr: u32, value: T) void { | ||||
|     const byte_addr = @truncate(u8, addr); | ||||
|  | ||||
|     switch (T) { | ||||
|         u32 => switch (byte_addr) { | ||||
|             0xB0 => dma.*[0].setDmasad(value), | ||||
|             0xB4 => dma.*[0].setDmadad(value), | ||||
|             0xB8 => dma.*[0].setDmacnt(value), | ||||
|  | ||||
|             0xBC => dma.*[1].setDmasad(value), | ||||
|             0xC0 => dma.*[1].setDmadad(value), | ||||
|             0xC4 => dma.*[1].setDmacnt(value), | ||||
|  | ||||
|             0xC8 => dma.*[2].setDmasad(value), | ||||
|             0xCC => dma.*[2].setDmadad(value), | ||||
|             0xD0 => dma.*[2].setDmacnt(value), | ||||
|  | ||||
|             0xD4 => dma.*[3].setDmasad(value), | ||||
|             0xD8 => dma.*[3].setDmadad(value), | ||||
|             0xDC => dma.*[3].setDmacnt(value), | ||||
|             else => util.io.write.undef(log, "Tried to write 0x{X:0>8}{} to 0x{X:0>8}", .{ value, T, addr }), | ||||
|         }, | ||||
|         u16 => switch (byte_addr) { | ||||
|             0xB0, 0xB2 => dma.*[0].setDmasad(setHalf(u32, dma.*[0].sad, byte_addr, value)), | ||||
|             0xB4, 0xB6 => dma.*[0].setDmadad(setHalf(u32, dma.*[0].dad, byte_addr, value)), | ||||
|             0xB8 => dma.*[0].setDmacntL(value), | ||||
|             0xBA => dma.*[0].setDmacntH(value), | ||||
|  | ||||
|             0xBC, 0xBE => dma.*[1].setDmasad(setHalf(u32, dma.*[1].sad, byte_addr, value)), | ||||
|             0xC0, 0xC2 => dma.*[1].setDmadad(setHalf(u32, dma.*[1].dad, byte_addr, value)), | ||||
|             0xC4 => dma.*[1].setDmacntL(value), | ||||
|             0xC6 => dma.*[1].setDmacntH(value), | ||||
|  | ||||
|             0xC8, 0xCA => dma.*[2].setDmasad(setHalf(u32, dma.*[2].sad, byte_addr, value)), | ||||
|             0xCC, 0xCE => dma.*[2].setDmadad(setHalf(u32, dma.*[2].dad, byte_addr, value)), | ||||
|             0xD0 => dma.*[2].setDmacntL(value), | ||||
|             0xD2 => dma.*[2].setDmacntH(value), | ||||
|  | ||||
|             0xD4, 0xD6 => dma.*[3].setDmasad(setHalf(u32, dma.*[3].sad, byte_addr, value)), | ||||
|             0xD8, 0xDA => dma.*[3].setDmadad(setHalf(u32, dma.*[3].dad, byte_addr, value)), | ||||
|             0xDC => dma.*[3].setDmacntL(value), | ||||
|             0xDE => dma.*[3].setDmacntH(value), | ||||
|             else => util.io.write.undef(log, "Tried to write 0x{X:0>4}{} to 0x{X:0>8}", .{ value, T, addr }), | ||||
|         }, | ||||
|         u8 => switch (byte_addr) { | ||||
|             0xB0, 0xB1, 0xB2, 0xB3 => dma.*[0].setDmasad(setQuart(dma.*[0].sad, byte_addr, value)), | ||||
|             0xB4, 0xB5, 0xB6, 0xB7 => dma.*[0].setDmadad(setQuart(dma.*[0].dad, byte_addr, value)), | ||||
|             0xB8, 0xB9 => dma.*[0].setDmacntL(setHalf(u16, dma.*[0].word_count, byte_addr, value)), | ||||
|             0xBA, 0xBB => dma.*[0].setDmacntH(setHalf(u16, dma.*[0].cnt.raw, byte_addr, value)), | ||||
|  | ||||
|             0xBC, 0xBD, 0xBE, 0xBF => dma.*[1].setDmasad(setQuart(dma.*[1].sad, byte_addr, value)), | ||||
|             0xC0, 0xC1, 0xC2, 0xC3 => dma.*[1].setDmadad(setQuart(dma.*[1].dad, byte_addr, value)), | ||||
|             0xC4, 0xC5 => dma.*[1].setDmacntL(setHalf(u16, dma.*[1].word_count, byte_addr, value)), | ||||
|             0xC6, 0xC7 => dma.*[1].setDmacntH(setHalf(u16, dma.*[1].cnt.raw, byte_addr, value)), | ||||
|  | ||||
|             0xC8, 0xC9, 0xCA, 0xCB => dma.*[2].setDmasad(setQuart(dma.*[2].sad, byte_addr, value)), | ||||
|             0xCC, 0xCD, 0xCE, 0xCF => dma.*[2].setDmadad(setQuart(dma.*[2].dad, byte_addr, value)), | ||||
|             0xD0, 0xD1 => dma.*[2].setDmacntL(setHalf(u16, dma.*[2].word_count, byte_addr, value)), | ||||
|             0xD2, 0xD3 => dma.*[2].setDmacntH(setHalf(u16, dma.*[2].cnt.raw, byte_addr, value)), | ||||
|  | ||||
|             0xD4, 0xD5, 0xD6, 0xD7 => dma.*[3].setDmasad(setQuart(dma.*[3].sad, byte_addr, value)), | ||||
|             0xD8, 0xD9, 0xDA, 0xDB => dma.*[3].setDmadad(setQuart(dma.*[3].dad, byte_addr, value)), | ||||
|             0xDC, 0xDD => dma.*[3].setDmacntL(setHalf(u16, dma.*[3].word_count, byte_addr, value)), | ||||
|             0xDE, 0xDF => dma.*[3].setDmacntH(setHalf(u16, dma.*[3].cnt.raw, byte_addr, value)), | ||||
|             else => util.io.write.undef(log, "Tried to write 0x{X:0>2}{} to 0x{X:0>8}", .{ value, T, addr }), | ||||
|         }, | ||||
|         else => @compileError("DMA: Unsupported write width"), | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Function that creates a DMAController. Determines unique DMA Controller behaiour at compile-time | ||||
| fn DmaController(comptime id: u2) type { | ||||
|     return struct { | ||||
|         const Self = @This(); | ||||
|  | ||||
|         const sad_mask: u32 = if (id == 0) 0x07FF_FFFF else 0x0FFF_FFFF; | ||||
|         const dad_mask: u32 = if (id != 3) 0x07FF_FFFF else 0x0FFF_FFFF; | ||||
|         const WordCount = if (id == 3) u16 else u14; | ||||
|  | ||||
|         /// Write-only. The first address in a DMA transfer. (DMASAD) | ||||
|         /// Note: use writeSrc instead of manipulating src_addr directly | ||||
|         sad: u32, | ||||
|         /// Write-only. The final address in a DMA transffer. (DMADAD) | ||||
|         /// Note: Use writeDst instead of manipulatig dst_addr directly | ||||
|         dad: u32, | ||||
|         /// Write-only. The Word Count for the DMA Transfer (DMACNT_L) | ||||
|         word_count: WordCount, | ||||
|         /// Read / Write. DMACNT_H | ||||
|         /// Note: Use writeControl instead of manipulating cnt directly. | ||||
|         cnt: DmaControl, | ||||
|  | ||||
|         /// Internal. The last successfully read value | ||||
|         data_latch: u32, | ||||
|         /// Internal. Currrent Source Address | ||||
|         sad_latch: u32, | ||||
|         /// Internal. Current Destination Address | ||||
|         dad_latch: u32, | ||||
|         /// Internal. Word Count | ||||
|         _word_count: WordCount, | ||||
|  | ||||
|         /// Some DMA Transfers are enabled during Hblank / VBlank and / or | ||||
|         /// have delays. Thefore bit 15 of DMACNT isn't actually something | ||||
|         /// we can use to control when we do or do not execute a step in a DMA Transfer | ||||
|         in_progress: bool, | ||||
|  | ||||
|         pub fn init() Self { | ||||
|             return .{ | ||||
|                 .sad = 0, | ||||
|                 .dad = 0, | ||||
|                 .word_count = 0, | ||||
|                 .cnt = .{ .raw = 0x000 }, | ||||
|  | ||||
|                 // Internals | ||||
|                 .sad_latch = 0, | ||||
|                 .dad_latch = 0, | ||||
|                 .data_latch = 0, | ||||
|  | ||||
|                 ._word_count = 0, | ||||
|                 .in_progress = false, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         pub fn setDmasad(self: *Self, addr: u32) void { | ||||
|             self.sad = addr & sad_mask; | ||||
|         } | ||||
|  | ||||
|         pub fn setDmadad(self: *Self, addr: u32) void { | ||||
|             self.dad = addr & dad_mask; | ||||
|         } | ||||
|  | ||||
|         pub fn setDmacntL(self: *Self, halfword: u16) void { | ||||
|             self.word_count = @truncate(@TypeOf(self.word_count), halfword); | ||||
|         } | ||||
|  | ||||
|         pub fn dmacntH(self: *const Self) u16 { | ||||
|             return self.cnt.raw & if (id == 3) 0xFFE0 else 0xF7E0; | ||||
|         } | ||||
|  | ||||
|         pub fn setDmacntH(self: *Self, halfword: u16) void { | ||||
|             const new = DmaControl{ .raw = halfword }; | ||||
|  | ||||
|             if (!self.cnt.enabled.read() and new.enabled.read()) { | ||||
|                 // Reload Internals on Rising Edge. | ||||
|                 self.sad_latch = self.sad; | ||||
|                 self.dad_latch = self.dad; | ||||
|                 self._word_count = if (self.word_count == 0) std.math.maxInt(WordCount) else self.word_count; | ||||
|  | ||||
|                 // Only a Start Timing of 00 has a DMA Transfer immediately begin | ||||
|                 self.in_progress = new.start_timing.read() == 0b00; | ||||
|             } | ||||
|  | ||||
|             self.cnt.raw = halfword; | ||||
|         } | ||||
|  | ||||
|         pub fn setDmacnt(self: *Self, word: u32) void { | ||||
|             self.setDmacntL(@truncate(u16, word)); | ||||
|             self.setDmacntH(@truncate(u16, word >> 16)); | ||||
|         } | ||||
|  | ||||
|         pub fn step(self: *Self, cpu: *Arm7tdmi) void { | ||||
|             const is_fifo = (id == 1 or id == 2) and self.cnt.start_timing.read() == 0b11; | ||||
|             const sad_adj = @intToEnum(Adjustment, self.cnt.sad_adj.read()); | ||||
|             const dad_adj = if (is_fifo) .Fixed else @intToEnum(Adjustment, self.cnt.dad_adj.read()); | ||||
|  | ||||
|             const transfer_type = is_fifo or self.cnt.transfer_type.read(); | ||||
|             const offset: u32 = if (transfer_type) @sizeOf(u32) else @sizeOf(u16); | ||||
|  | ||||
|             const mask = if (transfer_type) ~@as(u32, 3) else ~@as(u32, 1); | ||||
|             const sad_addr = self.sad_latch & mask; | ||||
|             const dad_addr = self.dad_latch & mask; | ||||
|  | ||||
|             if (transfer_type) { | ||||
|                 if (sad_addr >= 0x0200_0000) self.data_latch = cpu.bus.read(u32, sad_addr); | ||||
|                 cpu.bus.write(u32, dad_addr, self.data_latch); | ||||
|             } else { | ||||
|                 if (sad_addr >= 0x0200_0000) { | ||||
|                     const value: u32 = cpu.bus.read(u16, sad_addr); | ||||
|                     self.data_latch = value << 16 | value; | ||||
|                 } | ||||
|  | ||||
|                 cpu.bus.write(u16, dad_addr, @truncate(u16, rotr(u32, self.data_latch, 8 * (dad_addr & 3)))); | ||||
|             } | ||||
|  | ||||
|             switch (@truncate(u8, sad_addr >> 24)) { | ||||
|                 // according to fleroviux, DMAs with a source address in ROM misbehave | ||||
|                 // the resultant behaviour is that the source address will increment despite what DMAXCNT says | ||||
|                 0x08...0x0D => self.sad_latch +%= offset, // obscure behaviour | ||||
|                 else => switch (sad_adj) { | ||||
|                     .Increment => self.sad_latch +%= offset, | ||||
|                     .Decrement => self.sad_latch -%= offset, | ||||
|                     .IncrementReload => log.err("{} is a prohibited adjustment on SAD", .{sad_adj}), | ||||
|                     .Fixed => {}, | ||||
|                 }, | ||||
|             } | ||||
|  | ||||
|             switch (dad_adj) { | ||||
|                 .Increment, .IncrementReload => self.dad_latch +%= offset, | ||||
|                 .Decrement => self.dad_latch -%= offset, | ||||
|                 .Fixed => {}, | ||||
|             } | ||||
|  | ||||
|             self._word_count -= 1; | ||||
|  | ||||
|             if (self._word_count == 0) { | ||||
|                 if (self.cnt.irq.read()) { | ||||
|                     switch (id) { | ||||
|                         0 => cpu.bus.io.irq.dma0.set(), | ||||
|                         1 => cpu.bus.io.irq.dma1.set(), | ||||
|                         2 => cpu.bus.io.irq.dma2.set(), | ||||
|                         3 => cpu.bus.io.irq.dma3.set(), | ||||
|                     } | ||||
|  | ||||
|                     cpu.handleInterrupt(); | ||||
|                 } | ||||
|  | ||||
|                 // If we're not repeating, Fire the IRQs and disable the DMA | ||||
|                 if (!self.cnt.repeat.read()) self.cnt.enabled.unset(); | ||||
|  | ||||
|                 // We want to disable our internal enabled flag regardless of repeat | ||||
|                 // because we only want to step A DMA that repeats during it's specific | ||||
|                 // timing window | ||||
|                 self.in_progress = false; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fn poll(self: *Self, comptime kind: DmaKind) void { | ||||
|             if (self.in_progress) return; // If there's an ongoing DMA Transfer, exit early | ||||
|  | ||||
|             // No ongoing DMA Transfer, We want to check if we should repeat an existing one | ||||
|             // Determined by the repeat bit and whether the DMA is in the right start_timing | ||||
|             switch (kind) { | ||||
|                 .VBlank => self.in_progress = self.cnt.enabled.read() and self.cnt.start_timing.read() == 0b01, | ||||
|                 .HBlank => self.in_progress = self.cnt.enabled.read() and self.cnt.start_timing.read() == 0b10, | ||||
|                 .Immediate, .Special => {}, | ||||
|             } | ||||
|  | ||||
|             // If we determined that the repeat bit is set (and now the Hblank / Vblank DMA is now in progress) | ||||
|             // Reload internal word count latch | ||||
|             // Reload internal DAD latch if we are in IncrementRelaod | ||||
|             if (self.in_progress) { | ||||
|                 self._word_count = if (self.word_count == 0) std.math.maxInt(@TypeOf(self._word_count)) else self.word_count; | ||||
|                 if (@intToEnum(Adjustment, self.cnt.dad_adj.read()) == .IncrementReload) self.dad_latch = self.dad; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         pub fn requestAudio(self: *Self, _: u32) void { | ||||
|             comptime std.debug.assert(id == 1 or id == 2); | ||||
|             if (self.in_progress) return; // APU must wait their turn | ||||
|  | ||||
|             // DMA May not be configured for handling DMAs | ||||
|             if (self.cnt.start_timing.read() != 0b11) return; | ||||
|  | ||||
|             // We Assume the Repeat Bit is Set | ||||
|             // We Assume that DAD is set to 0x0400_00A0 or 0x0400_00A4 (fifo_addr) | ||||
|             // We Assume DMACNT_L is set to 4 | ||||
|  | ||||
|             // FIXME: Safe to just assume whatever DAD is set to is the FIFO Address? | ||||
|             // self.dad_latch = fifo_addr; | ||||
|             self.cnt.repeat.set(); | ||||
|             self._word_count = 4; | ||||
|             self.in_progress = true; | ||||
|         } | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn pollDmaOnBlank(bus: *Bus, comptime kind: DmaKind) void { | ||||
|     comptime var i: usize = 0; | ||||
|     inline while (i < 4) : (i += 1) { | ||||
|         bus.dma[i].poll(kind); | ||||
|     } | ||||
| } | ||||
|  | ||||
| const Adjustment = enum(u2) { | ||||
|     Increment = 0, | ||||
|     Decrement = 1, | ||||
|     Fixed = 2, | ||||
|     IncrementReload = 3, | ||||
| }; | ||||
|  | ||||
| const DmaKind = enum(u2) { | ||||
|     Immediate = 0, | ||||
|     HBlank, | ||||
|     VBlank, | ||||
|     Special, | ||||
| }; | ||||
| @@ -1,464 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const Bit = @import("bitfield").Bit; | ||||
| const DateTime = @import("datetime").datetime.Datetime; | ||||
|  | ||||
| const Arm7tdmi = @import("../cpu.zig").Arm7tdmi; | ||||
| const Allocator = std.mem.Allocator; | ||||
|  | ||||
| /// GPIO Register Implementation | ||||
| pub const Gpio = struct { | ||||
|     const Self = @This(); | ||||
|     const log = std.log.scoped(.Gpio); | ||||
|  | ||||
|     data: u4, | ||||
|     direction: u4, | ||||
|     cnt: u1, | ||||
|  | ||||
|     device: Device, | ||||
|  | ||||
|     const Register = enum { Data, Direction, Control }; | ||||
|  | ||||
|     pub const Device = struct { | ||||
|         ptr: ?*anyopaque, | ||||
|         kind: Kind, // TODO: Make comptime known? | ||||
|  | ||||
|         pub const Kind = enum { Rtc, None }; | ||||
|  | ||||
|         fn step(self: *Device, value: u4) u4 { | ||||
|             return switch (self.kind) { | ||||
|                 .Rtc => blk: { | ||||
|                     const clock = @ptrCast(*Clock, @alignCast(@alignOf(*Clock), self.ptr.?)); | ||||
|                     break :blk clock.step(Clock.Data{ .raw = value }); | ||||
|                 }, | ||||
|                 .None => value, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         fn init(kind: Kind, ptr: ?*anyopaque) Device { | ||||
|             return .{ .kind = kind, .ptr = ptr }; | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     pub fn write(self: *Self, comptime reg: Register, value: if (reg == .Control) u1 else u4) void { | ||||
|         switch (reg) { | ||||
|             .Data => { | ||||
|                 const masked_value = value & self.direction; | ||||
|  | ||||
|                 // The value which is actually stored in the GPIO register | ||||
|                 // might be modified by the device implementing the GPIO interface e.g. RTC reads | ||||
|                 self.data = self.device.step(masked_value); | ||||
|             }, | ||||
|             .Direction => self.direction = value, | ||||
|             .Control => self.cnt = value, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn read(self: *const Self, comptime reg: Register) if (reg == .Control) u1 else u4 { | ||||
|         if (self.cnt == 0) return 0; | ||||
|  | ||||
|         return switch (reg) { | ||||
|             .Data => self.data & ~self.direction, | ||||
|             .Direction => self.direction, | ||||
|             .Control => self.cnt, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn init(allocator: Allocator, cpu: *Arm7tdmi, kind: Device.Kind) !*Self { | ||||
|         log.info("Device: {}", .{kind}); | ||||
|  | ||||
|         const self = try allocator.create(Self); | ||||
|         errdefer allocator.destroy(self); | ||||
|  | ||||
|         self.* = .{ | ||||
|             .data = 0b0000, | ||||
|             .direction = 0b1111, // TODO: What is GPIO Direction set to by default? | ||||
|             .cnt = 0b0, | ||||
|  | ||||
|             .device = switch (kind) { | ||||
|                 .Rtc => blk: { | ||||
|                     const clock = try allocator.create(Clock); | ||||
|                     clock.init(cpu, self); | ||||
|  | ||||
|                     break :blk Device{ .kind = kind, .ptr = clock }; | ||||
|                 }, | ||||
|                 .None => Device{ .kind = kind, .ptr = null }, | ||||
|             }, | ||||
|         }; | ||||
|  | ||||
|         return self; | ||||
|     } | ||||
|  | ||||
|     pub fn deinit(self: *Self, allocator: Allocator) void { | ||||
|         switch (self.device.kind) { | ||||
|             .Rtc => allocator.destroy(@ptrCast(*Clock, @alignCast(@alignOf(*Clock), self.device.ptr.?))), | ||||
|             .None => {}, | ||||
|         } | ||||
|  | ||||
|         self.* = undefined; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| /// GBA Real Time Clock | ||||
| pub const Clock = struct { | ||||
|     const Self = @This(); | ||||
|     const log = std.log.scoped(.Rtc); | ||||
|  | ||||
|     writer: Writer, | ||||
|     reader: Reader, | ||||
|     state: State, | ||||
|     cnt: Control, | ||||
|  | ||||
|     year: u8, | ||||
|     month: u5, | ||||
|     day: u6, | ||||
|     weekday: u3, | ||||
|     hour: u6, | ||||
|     minute: u7, | ||||
|     second: u7, | ||||
|  | ||||
|     cpu: *Arm7tdmi, | ||||
|     gpio: *const Gpio, | ||||
|  | ||||
|     const Register = enum { | ||||
|         Control, | ||||
|         DateTime, | ||||
|         Time, | ||||
|     }; | ||||
|  | ||||
|     const State = union(enum) { | ||||
|         Idle, | ||||
|         Command, | ||||
|         Write: Register, | ||||
|         Read: Register, | ||||
|     }; | ||||
|  | ||||
|     const Reader = struct { | ||||
|         i: u4, | ||||
|         count: u8, | ||||
|  | ||||
|         /// Reads a bit from RTC registers. Which bit it reads is dependent on | ||||
|         /// | ||||
|         /// 1. The RTC State Machine, whitch tells us which register we're accessing | ||||
|         /// 2. A `count`, which keeps track of which byte is currently being read | ||||
|         /// 3. An index, which keeps track of which bit of the byte determined by `count` is being read | ||||
|         fn read(self: *Reader, clock: *const Clock, register: Register) u1 { | ||||
|             const idx = @intCast(u3, self.i); | ||||
|             defer self.i += 1; | ||||
|  | ||||
|             // FIXME: What do I do about the unused bits? | ||||
|             return switch (register) { | ||||
|                 .Control => @truncate(u1, switch (self.count) { | ||||
|                     0 => clock.cnt.raw >> idx, | ||||
|                     else => std.debug.panic("Tried to read from byte #{} of {} (hint: there's only 1 byte)", .{ self.count, register }), | ||||
|                 }), | ||||
|                 .DateTime => @truncate(u1, switch (self.count) { | ||||
|                     // Date | ||||
|                     0 => clock.year >> idx, | ||||
|                     1 => @as(u8, clock.month) >> idx, | ||||
|                     2 => @as(u8, clock.day) >> idx, | ||||
|                     3 => @as(u8, clock.weekday) >> idx, | ||||
|  | ||||
|                     // Time | ||||
|                     4 => @as(u8, clock.hour) >> idx, | ||||
|                     5 => @as(u8, clock.minute) >> idx, | ||||
|                     6 => @as(u8, clock.second) >> idx, | ||||
|                     else => std.debug.panic("Tried to read from byte #{} of {} (hint: there's only 7 bytes)", .{ self.count, register }), | ||||
|                 }), | ||||
|                 .Time => @truncate(u1, switch (self.count) { | ||||
|                     0 => @as(u8, clock.hour) >> idx, | ||||
|                     1 => @as(u8, clock.minute) >> idx, | ||||
|                     2 => @as(u8, clock.second) >> idx, | ||||
|                     else => std.debug.panic("Tried to read from byte #{} of {} (hint: there's only 3 bytes)", .{ self.count, register }), | ||||
|                 }), | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         /// Is true when a Reader has read a u8's worth of bits | ||||
|         fn finished(self: *const Reader) bool { | ||||
|             return self.i >= 8; | ||||
|         } | ||||
|  | ||||
|         /// Resets the index used to shift bits out of RTC registers | ||||
|         /// and `count`, which is used to keep track of which byte we're reading | ||||
|         /// is incremeneted | ||||
|         fn lap(self: *Reader) void { | ||||
|             self.i = 0; | ||||
|             self.count += 1; | ||||
|         } | ||||
|  | ||||
|         /// Resets the state of a `Reader` in preparation for a future | ||||
|         /// read command | ||||
|         fn reset(self: *Reader) void { | ||||
|             self.i = 0; | ||||
|             self.count = 0; | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     const Writer = struct { | ||||
|         buf: u8, | ||||
|         i: u4, | ||||
|  | ||||
|         /// The Number of bytes written since last reset | ||||
|         count: u8, | ||||
|  | ||||
|         /// Append a bit to the internal bit buffer (aka an integer) | ||||
|         fn push(self: *Writer, value: u1) void { | ||||
|             const idx = @intCast(u3, self.i); | ||||
|             self.buf = (self.buf & ~(@as(u8, 1) << idx)) | @as(u8, value) << idx; | ||||
|             self.i += 1; | ||||
|         } | ||||
|  | ||||
|         /// Takes the contents of the internal buffer and writes it to an RTC register | ||||
|         /// Where it writes to is dependent on: | ||||
|         /// | ||||
|         /// 1. The RTC State Machine, whitch tells us which register we're accessing | ||||
|         /// 2. A `count`, which keeps track of which byte is currently being read | ||||
|         fn write(self: *const Writer, clock: *Clock, register: Register) void { | ||||
|             // FIXME: What do do about unused bits? | ||||
|             switch (register) { | ||||
|                 .Control => switch (self.count) { | ||||
|                     0 => clock.cnt.raw = (clock.cnt.raw & 0x80) | (self.buf & 0x7F), // Bit 7 read-only | ||||
|                     else => std.debug.panic("Tried to write to byte #{} of {} (hint: there's only 1 byte)", .{ self.count, register }), | ||||
|                 }, | ||||
|                 .DateTime, .Time => log.debug("Ignoring {} write", .{register}), | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// Is true when 8 bits have been shifted into the internal buffer | ||||
|         fn finished(self: *const Writer) bool { | ||||
|             return self.i >= 8; | ||||
|         } | ||||
|  | ||||
|         /// Resets the internal buffer | ||||
|         /// resets the index used to shift bits into the internal buffer | ||||
|         /// increments `count` (which keeps track of byte offsets) by one | ||||
|         fn lap(self: *Writer) void { | ||||
|             self.buf = 0; | ||||
|             self.i = 0; | ||||
|             self.count += 1; | ||||
|         } | ||||
|  | ||||
|         /// Resets `Writer` to a clean state in preparation for a future write command | ||||
|         fn reset(self: *Writer) void { | ||||
|             self.buf = 0; | ||||
|             self.i = 0; | ||||
|             self.count = 0; | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     const Data = extern union { | ||||
|         sck: Bit(u8, 0), | ||||
|         sio: Bit(u8, 1), | ||||
|         cs: Bit(u8, 2), | ||||
|         raw: u8, | ||||
|     }; | ||||
|  | ||||
|     const Control = extern union { | ||||
|         /// Unknown, value should be preserved though | ||||
|         unk: Bit(u8, 1), | ||||
|         /// Per-minute IRQ | ||||
|         /// If set, fire a Gamepak IRQ every 30s, | ||||
|         irq: Bit(u8, 3), | ||||
|         /// 12/24 Hour Bit | ||||
|         /// If set, 12h mode | ||||
|         /// If cleared, 24h mode | ||||
|         mode: Bit(u8, 6), | ||||
|         /// Read-Only, bit cleared on read | ||||
|         /// If is set, means that there has been a failure / time has been lost | ||||
|         off: Bit(u8, 7), | ||||
|         raw: u8, | ||||
|     }; | ||||
|  | ||||
|     fn init(ptr: *Self, cpu: *Arm7tdmi, gpio: *const Gpio) void { | ||||
|         ptr.* = .{ | ||||
|             .writer = .{ .buf = 0, .i = 0, .count = 0 }, | ||||
|             .reader = .{ .i = 0, .count = 0 }, | ||||
|             .state = .Idle, | ||||
|             .cnt = .{ .raw = 0 }, | ||||
|             .year = 0x01, | ||||
|             .month = 0x6, | ||||
|             .day = 0x13, | ||||
|             .weekday = 0x3, | ||||
|             .hour = 0x23, | ||||
|             .minute = 0x59, | ||||
|             .second = 0x59, | ||||
|             .cpu = cpu, | ||||
|             .gpio = gpio, // Can't use Arm7tdmi ptr b/c not initialized yet | ||||
|         }; | ||||
|  | ||||
|         cpu.sched.push(.RealTimeClock, 1 << 24); // Every Second | ||||
|     } | ||||
|  | ||||
|     pub fn onClockUpdate(self: *Self, late: u64) void { | ||||
|         self.cpu.sched.push(.RealTimeClock, (1 << 24) -| late); // Reschedule | ||||
|  | ||||
|         const now = DateTime.now(); | ||||
|         self.year = bcd(u8, @intCast(u8, now.date.year - 2000)); | ||||
|         self.month = bcd(u5, now.date.month); | ||||
|         self.day = bcd(u6, now.date.day); | ||||
|         self.weekday = bcd(u3, (now.date.weekday() + 1) % 7); // API is Monday = 0, Sunday = 6. We want Sunday = 0, Saturday = 6 | ||||
|         self.hour = bcd(u6, now.time.hour); | ||||
|         self.minute = bcd(u7, now.time.minute); | ||||
|         self.second = bcd(u7, now.time.second); | ||||
|     } | ||||
|  | ||||
|     fn step(self: *Self, value: Data) u4 { | ||||
|         const cache: Data = .{ .raw = self.gpio.data }; | ||||
|  | ||||
|         return switch (self.state) { | ||||
|             .Idle => blk: { | ||||
|                 // FIXME: Maybe check incoming value to see if SCK is also high? | ||||
|                 if (cache.sck.read()) { | ||||
|                     if (!cache.cs.read() and value.cs.read()) { | ||||
|                         log.debug("Entering Command Mode", .{}); | ||||
|                         self.state = .Command; | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 break :blk @truncate(u4, value.raw); | ||||
|             }, | ||||
|             .Command => blk: { | ||||
|                 if (!value.cs.read()) log.err("Expected CS to be set during {}, however CS was cleared", .{self.state}); | ||||
|  | ||||
|                 // If SCK rises, sample SIO | ||||
|                 if (!cache.sck.read() and value.sck.read()) { | ||||
|                     self.writer.push(@boolToInt(value.sio.read())); | ||||
|  | ||||
|                     if (self.writer.finished()) { | ||||
|                         self.state = self.processCommand(self.writer.buf); | ||||
|                         self.writer.reset(); | ||||
|  | ||||
|                         log.debug("Switching to {}", .{self.state}); | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 break :blk @truncate(u4, value.raw); | ||||
|             }, | ||||
|             .Write => |register| blk: { | ||||
|                 if (!value.cs.read()) log.err("Expected CS to be set during {}, however CS was cleared", .{self.state}); | ||||
|  | ||||
|                 // If SCK rises, sample SIO | ||||
|                 if (!cache.sck.read() and value.sck.read()) { | ||||
|                     self.writer.push(@boolToInt(value.sio.read())); | ||||
|  | ||||
|                     const register_width: u32 = switch (register) { | ||||
|                         .Control => 1, | ||||
|                         .DateTime => 7, | ||||
|                         .Time => 3, | ||||
|                     }; | ||||
|  | ||||
|                     if (self.writer.finished()) { | ||||
|                         self.writer.write(self, register); // write inner buffer to RTC register | ||||
|                         self.writer.lap(); | ||||
|  | ||||
|                         if (self.writer.count == register_width) { | ||||
|                             self.writer.reset(); | ||||
|                             self.state = .Idle; | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 break :blk @truncate(u4, value.raw); | ||||
|             }, | ||||
|             .Read => |register| blk: { | ||||
|                 if (!value.cs.read()) log.err("Expected CS to be set during {}, however CS was cleared", .{self.state}); | ||||
|                 var ret = value; | ||||
|  | ||||
|                 // if SCK rises, sample SIO | ||||
|                 if (!cache.sck.read() and value.sck.read()) { | ||||
|                     ret.sio.write(self.reader.read(self, register) == 0b1); | ||||
|  | ||||
|                     const register_width: u32 = switch (register) { | ||||
|                         .Control => 1, | ||||
|                         .DateTime => 7, | ||||
|                         .Time => 3, | ||||
|                     }; | ||||
|  | ||||
|                     if (self.reader.finished()) { | ||||
|                         self.reader.lap(); | ||||
|  | ||||
|                         if (self.reader.count == register_width) { | ||||
|                             self.reader.reset(); | ||||
|                             self.state = .Idle; | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 break :blk @truncate(u4, ret.raw); | ||||
|             }, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn reset(self: *Self) void { | ||||
|         // mGBA and NBA only zero the control register. We will do the same | ||||
|         log.debug("Reset (control register was zeroed)", .{}); | ||||
|  | ||||
|         self.cnt.raw = 0; | ||||
|     } | ||||
|  | ||||
|     fn irq(self: *Self) void { | ||||
|         // TODO: Confirm that this is the right behaviour | ||||
|         log.debug("Force GamePak IRQ", .{}); | ||||
|  | ||||
|         self.cpu.bus.io.irq.game_pak.set(); | ||||
|         self.cpu.handleInterrupt(); | ||||
|     } | ||||
|  | ||||
|     fn processCommand(self: *Self, raw_command: u8) State { | ||||
|         const command = blk: { | ||||
|             // If High Nybble is 0x6, no need to switch the endianness | ||||
|             if (raw_command >> 4 & 0xF == 0x6) break :blk raw_command; | ||||
|  | ||||
|             // Turns out reversing the order of bits isn't trivial at all | ||||
|             // https://stackoverflow.com/questions/2602823/in-c-c-whats-the-simplest-way-to-reverse-the-order-of-bits-in-a-byte | ||||
|             var ret = raw_command; | ||||
|             ret = (ret & 0xF0) >> 4 | (ret & 0x0F) << 4; | ||||
|             ret = (ret & 0xCC) >> 2 | (ret & 0x33) << 2; | ||||
|             ret = (ret & 0xAA) >> 1 | (ret & 0x55) << 1; | ||||
|  | ||||
|             break :blk ret; | ||||
|         }; | ||||
|         log.debug("Handling Command 0x{X:0>2} [0b{b:0>8}]", .{ command, command }); | ||||
|  | ||||
|         const is_write = command & 1 == 0; | ||||
|         const rtc_register = @truncate(u3, command >> 1 & 0x7); | ||||
|  | ||||
|         if (is_write) { | ||||
|             return switch (rtc_register) { | ||||
|                 0 => blk: { | ||||
|                     self.reset(); | ||||
|                     break :blk .Idle; | ||||
|                 }, | ||||
|                 1 => .{ .Write = .Control }, | ||||
|                 2 => .{ .Write = .DateTime }, | ||||
|                 3 => .{ .Write = .Time }, | ||||
|                 6 => blk: { | ||||
|                     self.irq(); | ||||
|                     break :blk .Idle; | ||||
|                 }, | ||||
|                 4, 5, 7 => .Idle, | ||||
|             }; | ||||
|         } else { | ||||
|             return switch (rtc_register) { | ||||
|                 1 => .{ .Read = .Control }, | ||||
|                 2 => .{ .Read = .DateTime }, | ||||
|                 3 => .{ .Read = .Time }, | ||||
|                 0, 4, 5, 6, 7 => .Idle, // Do Nothing | ||||
|             }; | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| fn bcd(comptime T: type, value: u8) T { | ||||
|     var input = value; | ||||
|     var ret: u8 = 0; | ||||
|     var shift: u3 = 0; | ||||
|  | ||||
|     while (input > 0) { | ||||
|         ret |= (input % 10) << (shift << 2); | ||||
|         shift += 1; | ||||
|         input /= 10; | ||||
|     } | ||||
|  | ||||
|     return @truncate(T, ret); | ||||
| } | ||||
| @@ -1,671 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const timer = @import("timer.zig"); | ||||
| const dma = @import("dma.zig"); | ||||
| const apu = @import("../apu.zig"); | ||||
| const ppu = @import("../ppu.zig"); | ||||
| const util = @import("../../util.zig"); | ||||
|  | ||||
| const Bit = @import("bitfield").Bit; | ||||
| const Bitfield = @import("bitfield").Bitfield; | ||||
| const Bus = @import("../Bus.zig"); | ||||
|  | ||||
| const getHalf = util.getHalf; | ||||
| const setHalf = util.setHalf; | ||||
|  | ||||
| const log = std.log.scoped(.@"I/O"); | ||||
|  | ||||
| pub const Io = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     /// Read / Write | ||||
|     ime: bool, | ||||
|     ie: InterruptEnable, | ||||
|     irq: InterruptRequest, | ||||
|     postflg: PostFlag, | ||||
|     waitcnt: WaitControl, | ||||
|     haltcnt: HaltControl, | ||||
|     keyinput: AtomicKeyInput, | ||||
|  | ||||
|     pub fn init() Self { | ||||
|         return .{ | ||||
|             .ime = false, | ||||
|             .ie = .{ .raw = 0x0000 }, | ||||
|             .irq = .{ .raw = 0x0000 }, | ||||
|             .keyinput = AtomicKeyInput.init(.{ .raw = 0x03FF }), | ||||
|             .waitcnt = .{ .raw = 0x0000_0000 }, // Bit 15 == 0 for GBA | ||||
|             .postflg = .FirstBoot, | ||||
|             .haltcnt = .Execute, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn setIrqs(self: *Io, word: u32) void { | ||||
|         self.ie.raw = @truncate(u16, word); | ||||
|         self.irq.raw &= ~@truncate(u16, word >> 16); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| pub fn read(bus: *const Bus, comptime T: type, address: u32) ?T { | ||||
|     return switch (T) { | ||||
|         u32 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000...0x0400_0054 => ppu.read(T, &bus.ppu, address), | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0060...0x0400_00A4 => apu.read(T, &bus.apu, address), | ||||
|  | ||||
|             // DMA Transfers | ||||
|             0x0400_00B0...0x0400_00DC => dma.read(T, &bus.dma, address), | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100...0x0400_010C => timer.read(T, &bus.tim, address), | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0128 => util.io.read.todo(log, "Read {} from SIOCNT and SIOMLT_SEND", .{T}), | ||||
|  | ||||
|             // Keypad Input | ||||
|             0x0400_0130 => util.io.read.todo(log, "Read {} from KEYINPUT", .{T}), | ||||
|  | ||||
|             // Serial Communication 2 | ||||
|             0x0400_0150 => util.io.read.todo(log, "Read {} from JOY_RECV", .{T}), | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200 => @as(u32, bus.io.irq.raw) << 16 | bus.io.ie.raw, | ||||
|             0x0400_0204 => bus.io.waitcnt.raw, | ||||
|             0x0400_0208 => @boolToInt(bus.io.ime), | ||||
|             0x0400_0300 => @enumToInt(bus.io.postflg), | ||||
|             else => util.io.read.undef(T, log, "Tried to perform a {} read to 0x{X:0>8}", .{ T, address }), | ||||
|         }, | ||||
|         u16 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000...0x0400_0054 => ppu.read(T, &bus.ppu, address), | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0060...0x0400_00A6 => apu.read(T, &bus.apu, address), | ||||
|  | ||||
|             // DMA Transfers | ||||
|             0x0400_00B0...0x0400_00DE => dma.read(T, &bus.dma, address), | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100...0x0400_010E => timer.read(T, &bus.tim, address), | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0128 => util.io.read.todo(log, "Read {} from SIOCNT", .{T}), | ||||
|  | ||||
|             // Keypad Input | ||||
|             0x0400_0130 => bus.io.keyinput.load(.Monotonic).raw, | ||||
|  | ||||
|             // Serial Communication 2 | ||||
|             0x0400_0134 => util.io.read.todo(log, "Read {} from RCNT", .{T}), | ||||
|             0x0400_0136 => 0x0000, | ||||
|             0x0400_0142 => 0x0000, | ||||
|             0x0400_015A => 0x0000, | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200 => bus.io.ie.raw, | ||||
|             0x0400_0202 => bus.io.irq.raw, | ||||
|             0x0400_0204 => bus.io.waitcnt.raw, | ||||
|             0x0400_0206 => 0x0000, | ||||
|             0x0400_0208 => @boolToInt(bus.io.ime), | ||||
|             0x0400_020A => 0x0000, | ||||
|             0x0400_0300 => @enumToInt(bus.io.postflg), | ||||
|             0x0400_0302 => 0x0000, | ||||
|             else => util.io.read.undef(T, log, "Tried to perform a {} read to 0x{X:0>8}", .{ T, address }), | ||||
|         }, | ||||
|         u8 => return switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000...0x0400_0055 => ppu.read(T, &bus.ppu, address), | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0060...0x0400_00A7 => apu.read(T, &bus.apu, address), | ||||
|  | ||||
|             // DMA Transfers | ||||
|             0x0400_00B0...0x0400_00DF => dma.read(T, &bus.dma, address), | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100...0x0400_010F => timer.read(T, &bus.tim, address), | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0128 => util.io.read.todo(log, "Read {} from SIOCNT_L", .{T}), | ||||
|  | ||||
|             // Keypad Input | ||||
|             0x0400_0130 => util.io.read.todo(log, "read {} from KEYINPUT_L", .{T}), | ||||
|  | ||||
|             // Serial Communication 2 | ||||
|             0x0400_0135 => util.io.read.todo(log, "Read {} from RCNT_H", .{T}), | ||||
|             0x0400_0136, 0x0400_0137 => 0x00, | ||||
|             0x0400_0142, 0x0400_0143 => 0x00, | ||||
|             0x0400_015A, 0x0400_015B => 0x00, | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200, 0x0400_0201 => @truncate(T, bus.io.ie.raw >> getHalf(@truncate(u8, address))), | ||||
|             0x0400_0202, 0x0400_0203 => @truncate(T, bus.io.irq.raw >> getHalf(@truncate(u8, address))), | ||||
|             0x0400_0204, 0x0400_0205 => @truncate(T, bus.io.waitcnt.raw >> getHalf(@truncate(u8, address))), | ||||
|             0x0400_0206, 0x0400_0207 => 0x00, | ||||
|             0x0400_0208, 0x0400_0209 => @truncate(T, @as(u16, @boolToInt(bus.io.ime)) >> getHalf(@truncate(u8, address))), | ||||
|             0x0400_020A, 0x0400_020B => 0x00, | ||||
|             0x0400_0300 => @enumToInt(bus.io.postflg), | ||||
|             0x0400_0301 => null, | ||||
|             0x0400_0302, 0x0400_0303 => 0x00, | ||||
|             else => util.io.read.undef(T, log, "Tried to perform a {} read to 0x{X:0>8}", .{ T, address }), | ||||
|         }, | ||||
|         else => @compileError("I/O: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(bus: *Bus, comptime T: type, address: u32, value: T) void { | ||||
|     return switch (T) { | ||||
|         u32 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000...0x0400_0054 => ppu.write(T, &bus.ppu, address, value), | ||||
|             0x0400_0058...0x0400_005C => {}, // Unused | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0060...0x0400_00A4 => apu.write(T, &bus.apu, address, value), | ||||
|             0x0400_00A8, 0x0400_00AC => {}, // Unused | ||||
|  | ||||
|             // DMA Transfers | ||||
|             0x0400_00B0...0x0400_00DC => dma.write(T, &bus.dma, address, value), | ||||
|             0x0400_00E0...0x0400_00FC => {}, // Unused | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100...0x0400_010C => timer.write(T, &bus.tim, address, value), | ||||
|             0x0400_0110...0x0400_011C => {}, // Unused | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0120 => log.debug("Wrote 0x{X:0>8} to SIODATA32/(SIOMULTI0 and SIOMULTI1)", .{value}), | ||||
|             0x0400_0124 => log.debug("Wrote 0x{X:0>8} to SIOMULTI2 and SIOMULTI3", .{value}), | ||||
|             0x0400_0128 => log.debug("Wrote 0x{X:0>8} to SIOCNT and SIOMLT_SEND/SIODATA8", .{value}), | ||||
|             0x0400_012C => {}, // Unused | ||||
|  | ||||
|             // Keypad Input | ||||
|             0x0400_0130 => log.debug("Wrote 0x{X:0>8} to KEYINPUT and KEYCNT", .{value}), | ||||
|             0x0400_0134 => log.debug("Wrote 0x{X:0>8} to RCNT and IR", .{value}), | ||||
|             0x0400_0138, 0x0400_013C => {}, // Unused | ||||
|  | ||||
|             // Serial Communication 2 | ||||
|             0x0400_0140 => log.debug("Wrote 0x{X:0>8} to JOYCNT", .{value}), | ||||
|             0x0400_0150 => log.debug("Wrote 0x{X:0>8} to JOY_RECV", .{value}), | ||||
|             0x0400_0154 => log.debug("Wrote 0x{X:0>8} to JOY_TRANS", .{value}), | ||||
|             0x0400_0158 => log.debug("Wrote 0x{X:0>8} to JOYSTAT (?)", .{value}), | ||||
|             0x0400_0144...0x0400_014C, 0x0400_015C => {}, // Unused | ||||
|             0x0400_0160...0x0400_01FC => {}, | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200 => bus.io.setIrqs(value), | ||||
|             0x0400_0204 => bus.io.waitcnt.set(@truncate(u16, value)), | ||||
|             0x0400_0208 => bus.io.ime = value & 1 == 1, | ||||
|             0x0400_0300 => { | ||||
|                 bus.io.postflg = @intToEnum(PostFlag, value & 1); | ||||
|                 bus.io.haltcnt = if (value >> 15 & 1 == 0) .Halt else @panic("TODO: Implement STOP"); | ||||
|             }, | ||||
|             else => util.io.write.undef(log, "Tried to write 0x{X:0>8}{} to 0x{X:0>8}", .{ value, T, address }), | ||||
|         }, | ||||
|         u16 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000...0x0400_0054 => ppu.write(T, &bus.ppu, address, value), | ||||
|             0x0400_0056 => {}, // Not used | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0060...0x0400_00A6 => apu.write(T, &bus.apu, address, value), | ||||
|  | ||||
|             // Dma Transfers | ||||
|             0x0400_00B0...0x0400_00DE => dma.write(T, &bus.dma, address, value), | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100...0x0400_010E => timer.write(T, &bus.tim, address, value), | ||||
|             0x0400_0114 => {}, | ||||
|             0x0400_0110 => {}, // Not Used, | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0120 => log.debug("Wrote 0x{X:0>4} to SIOMULTI0", .{value}), | ||||
|             0x0400_0122 => log.debug("Wrote 0x{X:0>4} to SIOMULTI1", .{value}), | ||||
|             0x0400_0124 => log.debug("Wrote 0x{X:0>4} to SIOMULTI2", .{value}), | ||||
|             0x0400_0126 => log.debug("Wrote 0x{X:0>4} to SIOMULTI3", .{value}), | ||||
|             0x0400_0128 => log.debug("Wrote 0x{X:0>4} to SIOCNT", .{value}), | ||||
|             0x0400_012A => log.debug("Wrote 0x{X:0>4} to SIOMLT_SEND", .{value}), | ||||
|  | ||||
|             // Keypad Input | ||||
|             0x0400_0130 => log.debug("Wrote 0x{X:0>4} to KEYINPUT. Ignored", .{value}), | ||||
|             0x0400_0132 => log.debug("Wrote 0x{X:0>4} to KEYCNT", .{value}), | ||||
|  | ||||
|             // Serial Communication 2 | ||||
|             0x0400_0134 => log.debug("Wrote 0x{X:0>4} to RCNT", .{value}), | ||||
|             0x0400_0140 => log.debug("Wrote 0x{X:0>4} to JOYCNT", .{value}), | ||||
|             0x0400_0158 => log.debug("Wrote 0x{X:0>4} to JOYSTAT", .{value}), | ||||
|             0x0400_0142, 0x0400_015A => {}, // Not Used | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200 => bus.io.ie.raw = value, | ||||
|             0x0400_0202 => bus.io.irq.raw &= ~value, | ||||
|             0x0400_0204 => bus.io.waitcnt.set(value), | ||||
|             0x0400_0206 => {}, | ||||
|             0x0400_0208 => bus.io.ime = value & 1 == 1, | ||||
|             0x0400_020A => {}, | ||||
|             0x0400_0300 => { | ||||
|                 bus.io.postflg = @intToEnum(PostFlag, value & 1); | ||||
|                 bus.io.haltcnt = if (value >> 15 & 1 == 0) .Halt else @panic("TODO: Implement STOP"); | ||||
|             }, | ||||
|             else => util.io.write.undef(log, "Tried to write 0x{X:0>4}{} to 0x{X:0>8}", .{ value, T, address }), | ||||
|         }, | ||||
|         u8 => switch (address) { | ||||
|             // Display | ||||
|             0x0400_0000...0x0400_0055 => ppu.write(T, &bus.ppu, address, value), | ||||
|  | ||||
|             // Sound | ||||
|             0x0400_0060...0x0400_00A7 => apu.write(T, &bus.apu, address, value), | ||||
|  | ||||
|             // Dma Transfers | ||||
|             0x0400_00B0...0x0400_00DF => dma.write(T, &bus.dma, address, value), | ||||
|  | ||||
|             // Timers | ||||
|             0x0400_0100...0x0400_010F => timer.write(T, &bus.tim, address, value), | ||||
|  | ||||
|             // Serial Communication 1 | ||||
|             0x0400_0120 => log.debug("Wrote 0x{X:0>2} to SIODATA32_L_L", .{value}), | ||||
|             0x0400_0128 => log.debug("Wrote 0x{X:0>2} to SIOCNT_L", .{value}), | ||||
|  | ||||
|             // Serial Communication 2 | ||||
|             0x0400_0135 => log.debug("Wrote 0x{X:0>2} to RCNT_H", .{value}), | ||||
|             0x0400_0140 => log.debug("Wrote 0x{X:0>2} to JOYCNT_L", .{value}), | ||||
|  | ||||
|             // Interrupts | ||||
|             0x0400_0200, 0x0400_0201 => bus.io.ie.raw = setHalf(u16, bus.io.ie.raw, @truncate(u8, address), value), | ||||
|             0x0400_0202 => bus.io.irq.raw &= ~@as(u16, value), | ||||
|             0x0400_0203 => bus.io.irq.raw &= ~@as(u16, value) << 8, // TODO: Is this good? | ||||
|             0x0400_0204, 0x0400_0205 => bus.io.waitcnt.set(setHalf(u16, @truncate(u16, bus.io.waitcnt.raw), @truncate(u8, address), value)), | ||||
|             0x0400_0206, 0x0400_0207 => {}, | ||||
|             0x0400_0208 => bus.io.ime = value & 1 == 1, | ||||
|             0x0400_0209 => {}, | ||||
|             0x0400_020A, 0x0400_020B => {}, | ||||
|  | ||||
|             0x0400_0300 => bus.io.postflg = @intToEnum(PostFlag, value & 1), | ||||
|             0x0400_0301 => bus.io.haltcnt = if (value >> 7 & 1 == 0) .Halt else std.debug.panic("TODO: Implement STOP", .{}), | ||||
|  | ||||
|             0x0400_0410 => log.debug("Wrote 0x{X:0>2} to the common yet undocumented 0x{X:0>8}", .{ value, address }), | ||||
|             else => util.io.write.undef(log, "Tried to write 0x{X:0>2}{} to 0x{X:0>8}", .{ value, T, address }), | ||||
|         }, | ||||
|         else => @compileError("I/O: Unsupported write width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| /// Read / Write | ||||
| pub const PostFlag = enum(u1) { | ||||
|     FirstBoot = 0, | ||||
|     FurtherBoots = 1, | ||||
| }; | ||||
|  | ||||
| /// Write Only | ||||
| pub const HaltControl = enum { | ||||
|     Halt, | ||||
|     Stop, | ||||
|     Execute, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const DisplayControl = extern union { | ||||
|     bg_mode: Bitfield(u16, 0, 3), | ||||
|     frame_select: Bit(u16, 4), | ||||
|     hblank_interval_free: Bit(u16, 5), | ||||
|     obj_mapping: Bit(u16, 6), | ||||
|     forced_blank: Bit(u16, 7), | ||||
|     bg_enable: Bitfield(u16, 8, 4), | ||||
|     obj_enable: Bit(u16, 12), | ||||
|     win_enable: Bitfield(u16, 13, 2), | ||||
|     obj_win_enable: Bit(u16, 15), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const DisplayStatus = extern union { | ||||
|     /// read-only | ||||
|     vblank: Bit(u16, 0), | ||||
|     /// read-only | ||||
|     hblank: Bit(u16, 1), | ||||
|     // read-only | ||||
|     coincidence: Bit(u16, 2), | ||||
|     vblank_irq: Bit(u16, 3), | ||||
|     hblank_irq: Bit(u16, 4), | ||||
|     vcount_irq: Bit(u16, 5), | ||||
|     vcount_trigger: Bitfield(u16, 8, 8), | ||||
|     raw: u16, | ||||
|  | ||||
|     pub fn set(self: *DisplayStatus, value: u16) void { | ||||
|         const mask: u16 = 0x00C7; // set bits are read-only | ||||
|         self.raw = (self.raw & mask) | (value & ~mask); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| /// Read Only | ||||
| pub const VCount = extern union { | ||||
|     scanline: Bitfield(u16, 0, 8), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| const InterruptEnable = extern union { | ||||
|     vblank: Bit(u16, 0), | ||||
|     hblank: Bit(u16, 1), | ||||
|     coincidence: Bit(u16, 2), | ||||
|     tm0_overflow: Bit(u16, 3), | ||||
|     tm1_overflow: Bit(u16, 4), | ||||
|     tm2_overflow: Bit(u16, 5), | ||||
|     tm3_overflow: Bit(u16, 6), | ||||
|     serial: Bit(u16, 7), | ||||
|     dma0: Bit(u16, 8), | ||||
|     dma1: Bit(u16, 9), | ||||
|     dma2: Bit(u16, 10), | ||||
|     dma3: Bit(u16, 11), | ||||
|     keypad: Bit(u16, 12), | ||||
|     game_pak: Bit(u16, 13), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read Only | ||||
| /// 0 = Pressed, 1 = Released | ||||
| const KeyInput = extern union { | ||||
|     a: Bit(u16, 0), | ||||
|     b: Bit(u16, 1), | ||||
|     select: Bit(u16, 2), | ||||
|     start: Bit(u16, 3), | ||||
|     right: Bit(u16, 4), | ||||
|     left: Bit(u16, 5), | ||||
|     up: Bit(u16, 6), | ||||
|     down: Bit(u16, 7), | ||||
|     shoulder_r: Bit(u16, 8), | ||||
|     shoulder_l: Bit(u16, 9), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| const AtomicKeyInput = struct { | ||||
|     const Self = @This(); | ||||
|     const Ordering = std.atomic.Ordering; | ||||
|  | ||||
|     inner: KeyInput, | ||||
|  | ||||
|     pub fn init(value: KeyInput) Self { | ||||
|         return .{ .inner = value }; | ||||
|     } | ||||
|  | ||||
|     pub inline fn load(self: *const Self, comptime ordering: Ordering) KeyInput { | ||||
|         return .{ .raw = switch (ordering) { | ||||
|             .AcqRel, .Release => @compileError("not supported for atomic loads"), | ||||
|             else => @atomicLoad(u16, &self.inner.raw, ordering), | ||||
|         } }; | ||||
|     } | ||||
|  | ||||
|     pub inline fn store(self: *Self, value: u16, comptime ordering: Ordering) void { | ||||
|         switch (ordering) { | ||||
|             .AcqRel, .Acquire => @compileError("not supported for atomic stores"), | ||||
|             else => @atomicStore(u16, &self.inner.raw, value, ordering), | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| // Read / Write | ||||
| pub const BackgroundControl = extern union { | ||||
|     priority: Bitfield(u16, 0, 2), | ||||
|     char_base: Bitfield(u16, 2, 2), | ||||
|     mosaic_enable: Bit(u16, 6), | ||||
|     colour_mode: Bit(u16, 7), | ||||
|     screen_base: Bitfield(u16, 8, 5), | ||||
|     display_overflow: Bit(u16, 13), | ||||
|     size: Bitfield(u16, 14, 2), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Write Only | ||||
| pub const BackgroundOffset = extern union { | ||||
|     offset: Bitfield(u16, 0, 9), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const BldCnt = extern union { | ||||
|     /// BLDCNT{0} is BG0 A | ||||
|     /// BLDCNT{4} is OBJ A | ||||
|     /// BLDCNT{5} is BD  A | ||||
|     layer_a: Bitfield(u16, 0, 6), | ||||
|     mode: Bitfield(u16, 6, 2), | ||||
|  | ||||
|     /// BLDCNT{8} is BG0 B | ||||
|     /// BLDCNT{12} is OBJ B | ||||
|     /// BLDCNT{13} is BD  B | ||||
|     layer_b: Bitfield(u16, 8, 6), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read-only? | ||||
| /// Alpha Blending Coefficients | ||||
| pub const BldAlpha = extern union { | ||||
|     eva: Bitfield(u16, 0, 5), | ||||
|     evb: Bitfield(u16, 8, 5), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Write-only? | ||||
| /// Brightness COefficients | ||||
| pub const BldY = extern union { | ||||
|     evy: Bitfield(u16, 0, 5), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Write-only | ||||
| pub const WinH = extern union { | ||||
|     x2: Bitfield(u16, 0, 8), | ||||
|     x1: Bitfield(u16, 8, 8), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Write-only | ||||
| pub const WinV = extern union { | ||||
|     y2: Bitfield(u16, 0, 8), | ||||
|     y1: Bitfield(u16, 8, 8), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| pub const WinIn = extern union { | ||||
|     w0_bg: Bitfield(u16, 0, 4), | ||||
|     w0_obj: Bit(u16, 4), | ||||
|     w0_colour: Bit(u16, 5), | ||||
|     w1_bg: Bitfield(u16, 8, 4), | ||||
|     w1_obj: Bit(u16, 12), | ||||
|     w1_colour: Bit(u16, 13), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| pub const WinOut = extern union { | ||||
|     out_bg: Bitfield(u16, 0, 4), | ||||
|     out_obj: Bit(u16, 4), | ||||
|     out_colour: Bit(u16, 5), | ||||
|     obj_bg: Bitfield(u16, 8, 4), | ||||
|     obj_obj: Bit(u16, 12), | ||||
|     obj_colour: Bit(u16, 13), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| const InterruptRequest = extern union { | ||||
|     vblank: Bit(u16, 0), | ||||
|     hblank: Bit(u16, 1), | ||||
|     coincidence: Bit(u16, 2), | ||||
|     tim0: Bit(u16, 3), | ||||
|     tim1: Bit(u16, 4), | ||||
|     tim2: Bit(u16, 5), | ||||
|     tim3: Bit(u16, 6), | ||||
|     serial: Bit(u16, 7), | ||||
|     dma0: Bit(u16, 8), | ||||
|     dma1: Bit(u16, 9), | ||||
|     dma2: Bit(u16, 10), | ||||
|     dma3: Bit(u16, 11), | ||||
|     keypad: Bit(u16, 12), | ||||
|     game_pak: Bit(u16, 13), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const DmaControl = extern union { | ||||
|     dad_adj: Bitfield(u16, 5, 2), | ||||
|     sad_adj: Bitfield(u16, 7, 2), | ||||
|     repeat: Bit(u16, 9), | ||||
|     transfer_type: Bit(u16, 10), | ||||
|     pak_drq: Bit(u16, 11), | ||||
|     start_timing: Bitfield(u16, 12, 2), | ||||
|     irq: Bit(u16, 14), | ||||
|     enabled: Bit(u16, 15), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const TimerControl = extern union { | ||||
|     frequency: Bitfield(u16, 0, 2), | ||||
|     cascade: Bit(u16, 2), | ||||
|     irq: Bit(u16, 6), | ||||
|     enabled: Bit(u16, 7), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR10 | ||||
| pub const Sweep = extern union { | ||||
|     shift: Bitfield(u8, 0, 3), | ||||
|     direction: Bit(u8, 3), | ||||
|     period: Bitfield(u8, 4, 3), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// This represents the Duty / Len | ||||
| /// NRx1 | ||||
| pub const Duty = extern union { | ||||
|     /// Write-only | ||||
|     /// Only used when bit 6 is set | ||||
|     length: Bitfield(u16, 0, 6), | ||||
|     pattern: Bitfield(u16, 6, 2), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NRx2 | ||||
| pub const Envelope = extern union { | ||||
|     period: Bitfield(u8, 0, 3), | ||||
|     direction: Bit(u8, 3), | ||||
|     init_vol: Bitfield(u8, 4, 4), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NRx3, NRx4 | ||||
| pub const Frequency = extern union { | ||||
|     /// Write-only | ||||
|     frequency: Bitfield(u16, 0, 11), | ||||
|     length_enable: Bit(u16, 14), | ||||
|     /// Write-only | ||||
|     trigger: Bit(u16, 15), | ||||
|  | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR30 | ||||
| pub const WaveSelect = extern union { | ||||
|     dimension: Bit(u8, 5), | ||||
|     bank: Bit(u8, 6), | ||||
|     enabled: Bit(u8, 7), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR32 | ||||
| pub const WaveVolume = extern union { | ||||
|     kind: Bitfield(u8, 5, 2), | ||||
|     force: Bit(u8, 7), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR43 | ||||
| pub const PolyCounter = extern union { | ||||
|     div_ratio: Bitfield(u8, 0, 3), | ||||
|     width: Bit(u8, 3), | ||||
|     shift: Bitfield(u8, 4, 4), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| /// NR44 | ||||
| pub const NoiseControl = extern union { | ||||
|     length_enable: Bit(u8, 6), | ||||
|     trigger: Bit(u8, 7), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const ChannelVolumeControl = extern union { | ||||
|     right_vol: Bitfield(u16, 0, 3), | ||||
|     left_vol: Bitfield(u16, 4, 3), | ||||
|     ch_right: Bitfield(u16, 8, 4), | ||||
|     ch_left: Bitfield(u16, 12, 4), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const DmaSoundControl = extern union { | ||||
|     ch_vol: Bitfield(u16, 0, 2), | ||||
|     chA_vol: Bit(u16, 2), | ||||
|     chB_vol: Bit(u16, 3), | ||||
|  | ||||
|     chA_right: Bit(u16, 8), | ||||
|     chA_left: Bit(u16, 9), | ||||
|     chA_timer: Bit(u16, 10), | ||||
|     /// Write only? | ||||
|     chA_reset: Bit(u16, 11), | ||||
|  | ||||
|     chB_right: Bit(u16, 12), | ||||
|     chB_left: Bit(u16, 13), | ||||
|     chB_timer: Bit(u16, 14), | ||||
|     /// Write only? | ||||
|     chB_reset: Bit(u16, 15), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const SoundControl = extern union { | ||||
|     /// Read-only | ||||
|     ch1_enable: Bit(u8, 0), | ||||
|     /// Read-only | ||||
|     ch2_enable: Bit(u8, 1), | ||||
|     /// Read-only | ||||
|     ch3_enable: Bit(u8, 2), | ||||
|     /// Read-only | ||||
|     ch4_enable: Bit(u8, 3), | ||||
|     apu_enable: Bit(u8, 7), | ||||
|     raw: u8, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const SoundBias = extern union { | ||||
|     level: Bitfield(u16, 1, 9), | ||||
|     sampling_cycle: Bitfield(u16, 14, 2), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| /// Read / Write | ||||
| pub const WaitControl = extern union { | ||||
|     sram_cnt: Bitfield(u16, 0, 2), | ||||
|     s0_first: Bitfield(u16, 2, 2), | ||||
|     s0_second: Bit(u16, 4), | ||||
|     s1_first: Bitfield(u16, 5, 2), | ||||
|     s1_second: Bit(u16, 7), | ||||
|     s2_first: Bitfield(u16, 8, 2), | ||||
|     s2_second: Bit(u16, 10), | ||||
|     phi_out: Bitfield(u16, 11, 2), | ||||
|  | ||||
|     prefetch_enable: Bit(u16, 14), | ||||
|     pak_kind: Bit(u16, 15), | ||||
|     raw: u16, | ||||
|  | ||||
|     pub fn set(self: *WaitControl, value: u16) void { | ||||
|         const mask: u16 = 0x8000; // set bits are read-only | ||||
|         self.raw = (self.raw & mask) | (value & ~mask); | ||||
|     } | ||||
| }; | ||||
| @@ -1,243 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const util = @import("../../util.zig"); | ||||
|  | ||||
| const TimerControl = @import("io.zig").TimerControl; | ||||
| const Scheduler = @import("../scheduler.zig").Scheduler; | ||||
| const Arm7tdmi = @import("../cpu.zig").Arm7tdmi; | ||||
|  | ||||
| pub const TimerTuple = struct { Timer(0), Timer(1), Timer(2), Timer(3) }; | ||||
| const log = std.log.scoped(.Timer); | ||||
|  | ||||
| const getHalf = util.getHalf; | ||||
| const setHalf = util.setHalf; | ||||
|  | ||||
| pub fn create(sched: *Scheduler) TimerTuple { | ||||
|     return .{ Timer(0).init(sched), Timer(1).init(sched), Timer(2).init(sched), Timer(3).init(sched) }; | ||||
| } | ||||
|  | ||||
| pub fn read(comptime T: type, tim: *const TimerTuple, addr: u32) ?T { | ||||
|     const nybble_addr = @truncate(u4, addr); | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => switch (nybble_addr) { | ||||
|             0x0 => @as(T, tim.*[0].cnt.raw) << 16 | tim.*[0].timcntL(), | ||||
|             0x4 => @as(T, tim.*[1].cnt.raw) << 16 | tim.*[1].timcntL(), | ||||
|             0x8 => @as(T, tim.*[2].cnt.raw) << 16 | tim.*[2].timcntL(), | ||||
|             0xC => @as(T, tim.*[3].cnt.raw) << 16 | tim.*[3].timcntL(), | ||||
|             else => util.io.read.err(T, log, "unaligned {} read from 0x{X:0>8}", .{ T, addr }), | ||||
|         }, | ||||
|         u16 => switch (nybble_addr) { | ||||
|             0x0 => tim.*[0].timcntL(), | ||||
|             0x2 => tim.*[0].cnt.raw, | ||||
|  | ||||
|             0x4 => tim.*[1].timcntL(), | ||||
|             0x6 => tim.*[1].cnt.raw, | ||||
|  | ||||
|             0x8 => tim.*[2].timcntL(), | ||||
|             0xA => tim.*[2].cnt.raw, | ||||
|  | ||||
|             0xC => tim.*[3].timcntL(), | ||||
|             0xE => tim.*[3].cnt.raw, | ||||
|             else => util.io.read.err(T, log, "unaligned {} read from 0x{X:0>8}", .{ T, addr }), | ||||
|         }, | ||||
|         u8 => switch (nybble_addr) { | ||||
|             0x0, 0x1 => @truncate(T, tim.*[0].timcntL() >> getHalf(nybble_addr)), | ||||
|             0x2, 0x3 => @truncate(T, tim.*[0].cnt.raw >> getHalf(nybble_addr)), | ||||
|  | ||||
|             0x4, 0x5 => @truncate(T, tim.*[1].timcntL() >> getHalf(nybble_addr)), | ||||
|             0x6, 0x7 => @truncate(T, tim.*[1].cnt.raw >> getHalf(nybble_addr)), | ||||
|  | ||||
|             0x8, 0x9 => @truncate(T, tim.*[2].timcntL() >> getHalf(nybble_addr)), | ||||
|             0xA, 0xB => @truncate(T, tim.*[2].cnt.raw >> getHalf(nybble_addr)), | ||||
|  | ||||
|             0xC, 0xD => @truncate(T, tim.*[3].timcntL() >> getHalf(nybble_addr)), | ||||
|             0xE, 0xF => @truncate(T, tim.*[3].cnt.raw >> getHalf(nybble_addr)), | ||||
|         }, | ||||
|         else => @compileError("TIM: Unsupported read width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub fn write(comptime T: type, tim: *TimerTuple, addr: u32, value: T) void { | ||||
|     const nybble_addr = @truncate(u4, addr); | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => switch (nybble_addr) { | ||||
|             0x0 => tim.*[0].setTimcnt(value), | ||||
|             0x4 => tim.*[1].setTimcnt(value), | ||||
|             0x8 => tim.*[2].setTimcnt(value), | ||||
|             0xC => tim.*[3].setTimcnt(value), | ||||
|             else => util.io.write.undef(log, "Tried to write 0x{X:0>8}{} to 0x{X:0>8}", .{ value, T, addr }), | ||||
|         }, | ||||
|         u16 => switch (nybble_addr) { | ||||
|             0x0 => tim.*[0].setTimcntL(value), | ||||
|             0x2 => tim.*[0].setTimcntH(value), | ||||
|  | ||||
|             0x4 => tim.*[1].setTimcntL(value), | ||||
|             0x6 => tim.*[1].setTimcntH(value), | ||||
|  | ||||
|             0x8 => tim.*[2].setTimcntL(value), | ||||
|             0xA => tim.*[2].setTimcntH(value), | ||||
|  | ||||
|             0xC => tim.*[3].setTimcntL(value), | ||||
|             0xE => tim.*[3].setTimcntH(value), | ||||
|             else => util.io.write.undef(log, "Tried to write 0x{X:0>4}{} to 0x{X:0>8}", .{ value, T, addr }), | ||||
|         }, | ||||
|         u8 => switch (nybble_addr) { | ||||
|             0x0, 0x1 => tim.*[0].setTimcntL(setHalf(u16, tim.*[0]._reload, nybble_addr, value)), | ||||
|             0x2, 0x3 => tim.*[0].setTimcntH(setHalf(u16, tim.*[0].cnt.raw, nybble_addr, value)), | ||||
|  | ||||
|             0x4, 0x5 => tim.*[1].setTimcntL(setHalf(u16, tim.*[1]._reload, nybble_addr, value)), | ||||
|             0x6, 0x7 => tim.*[1].setTimcntH(setHalf(u16, tim.*[1].cnt.raw, nybble_addr, value)), | ||||
|  | ||||
|             0x8, 0x9 => tim.*[2].setTimcntL(setHalf(u16, tim.*[2]._reload, nybble_addr, value)), | ||||
|             0xA, 0xB => tim.*[2].setTimcntH(setHalf(u16, tim.*[2].cnt.raw, nybble_addr, value)), | ||||
|  | ||||
|             0xC, 0xD => tim.*[3].setTimcntL(setHalf(u16, tim.*[3]._reload, nybble_addr, value)), | ||||
|             0xE, 0xF => tim.*[3].setTimcntH(setHalf(u16, tim.*[3].cnt.raw, nybble_addr, value)), | ||||
|         }, | ||||
|         else => @compileError("TIM: Unsupported write width"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| fn Timer(comptime id: u2) type { | ||||
|     return struct { | ||||
|         const Self = @This(); | ||||
|  | ||||
|         /// Read Only, Internal. Please use self.timcntL() | ||||
|         _counter: u16, | ||||
|  | ||||
|         /// Write Only, Internal. Please use self.setTimcntL() | ||||
|         _reload: u16, | ||||
|  | ||||
|         /// Write Only, Internal. Please use self.setTimcntH() | ||||
|         cnt: TimerControl, | ||||
|  | ||||
|         /// Internal. | ||||
|         sched: *Scheduler, | ||||
|  | ||||
|         /// Internal | ||||
|         _start_timestamp: u64, | ||||
|  | ||||
|         pub fn init(sched: *Scheduler) Self { | ||||
|             return .{ | ||||
|                 ._reload = 0, | ||||
|                 ._counter = 0, | ||||
|                 .cnt = .{ .raw = 0x0000 }, | ||||
|                 .sched = sched, | ||||
|                 ._start_timestamp = 0, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         /// TIMCNT_L Getter | ||||
|         pub fn timcntL(self: *const Self) u16 { | ||||
|             if (self.cnt.cascade.read() or !self.cnt.enabled.read()) return self._counter; | ||||
|  | ||||
|             return self._counter +% @truncate(u16, (self.sched.now() - self._start_timestamp) / self.frequency()); | ||||
|         } | ||||
|  | ||||
|         /// TIMCNT_L Setter | ||||
|         pub fn setTimcntL(self: *Self, halfword: u16) void { | ||||
|             self._reload = halfword; | ||||
|         } | ||||
|  | ||||
|         /// TIMCNT_L & TIMCNT_H | ||||
|         pub fn setTimcnt(self: *Self, word: u32) void { | ||||
|             self.setTimcntL(@truncate(u16, word)); | ||||
|             self.setTimcntH(@truncate(u16, word >> 16)); | ||||
|         } | ||||
|  | ||||
|         /// TIMCNT_H | ||||
|         pub fn setTimcntH(self: *Self, halfword: u16) void { | ||||
|             const new = TimerControl{ .raw = halfword }; | ||||
|  | ||||
|             if (self.cnt.enabled.read()) { | ||||
|                 // timer was already enabled | ||||
|  | ||||
|                 // If enabled falling edge or cascade falling edge, timer is paused | ||||
|                 if (!new.enabled.read() or (!self.cnt.cascade.read() and new.cascade.read())) { | ||||
|                     self.sched.removeScheduledEvent(.{ .TimerOverflow = id }); | ||||
|  | ||||
|                     // Counter should hold the value it stopped at meaning we have to calculate it now | ||||
|                     self._counter +%= @truncate(u16, (self.sched.now() - self._start_timestamp) / self.frequency()); | ||||
|                 } | ||||
|  | ||||
|                 // the timer has always been enabled, but the cascade bit which was blocking the timer has been unset | ||||
|                 if (new.enabled.read() and (self.cnt.cascade.read() and !new.cascade.read())) { | ||||
|                     // we want to reschedule the timer event, however we won't reload the counter. | ||||
|                     // the invariant here is that self._counter holds the already calculated paused value | ||||
|  | ||||
|                     self.rescheduleTimerExpire(0); | ||||
|                 } | ||||
|             } else { | ||||
|                 // the timer was previously disabeld | ||||
|  | ||||
|                 if (new.enabled.read()) { | ||||
|                     // timer should start counting (with a reloaded counter value) | ||||
|                     self._counter = self._reload; | ||||
|  | ||||
|                     // if cascade happens to be set, the timer doesn't actually do anything though | ||||
|                     if (!new.cascade.read()) self.rescheduleTimerExpire(0); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             self.cnt.raw = halfword; | ||||
|         } | ||||
|  | ||||
|         pub fn onTimerExpire(self: *Self, cpu: *Arm7tdmi, late: u64) void { | ||||
|             // Fire IRQ if enabled | ||||
|             const io = &cpu.bus.io; | ||||
|  | ||||
|             if (self.cnt.irq.read()) { | ||||
|                 switch (id) { | ||||
|                     0 => io.irq.tim0.set(), | ||||
|                     1 => io.irq.tim1.set(), | ||||
|                     2 => io.irq.tim2.set(), | ||||
|                     3 => io.irq.tim3.set(), | ||||
|                 } | ||||
|  | ||||
|                 cpu.handleInterrupt(); | ||||
|             } | ||||
|  | ||||
|             // DMA Sound Things | ||||
|             if (id == 0 or id == 1) { | ||||
|                 cpu.bus.apu.onDmaAudioSampleRequest(cpu, id); | ||||
|             } | ||||
|  | ||||
|             // Perform Cascade Behaviour | ||||
|             switch (id) { | ||||
|                 inline 0, 1, 2 => |idx| { | ||||
|                     const next = idx + 1; | ||||
|  | ||||
|                     if (cpu.bus.tim[next].cnt.cascade.read()) { | ||||
|                         cpu.bus.tim[next]._counter +%= 1; | ||||
|                         if (cpu.bus.tim[next]._counter == 0) cpu.bus.tim[next].onTimerExpire(cpu, late); | ||||
|                     } | ||||
|                 }, | ||||
|                 3 => {}, // THere is no timer for TIM3 to cascade to | ||||
|             } | ||||
|  | ||||
|             // Reschedule Timer if we're not cascading | ||||
|             // TIM0 cascade value is N/A | ||||
|             if (id == 0 or !self.cnt.cascade.read()) { | ||||
|                 self._counter = self._reload; | ||||
|                 self.rescheduleTimerExpire(late); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fn rescheduleTimerExpire(self: *Self, late: u64) void { | ||||
|             const when = (@as(u64, 0x10000) - self._counter) * self.frequency(); | ||||
|  | ||||
|             self._start_timestamp = self.sched.now(); | ||||
|             self.sched.push(.{ .TimerOverflow = id }, when -| late); | ||||
|         } | ||||
|  | ||||
|         fn frequency(self: *const Self) u16 { | ||||
|             return switch (self.cnt.frequency.read()) { | ||||
|                 0 => 1, | ||||
|                 1 => 64, | ||||
|                 2 => 256, | ||||
|                 3 => 1024, | ||||
|             }; | ||||
|         } | ||||
|     }; | ||||
| } | ||||
							
								
								
									
										678
									
								
								src/core/cpu.zig
									
									
									
									
									
								
							
							
						
						
									
										678
									
								
								src/core/cpu.zig
									
									
									
									
									
								
							| @@ -1,678 +0,0 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Bus = @import("Bus.zig"); | ||||
| const Bit = @import("bitfield").Bit; | ||||
| const Bitfield = @import("bitfield").Bitfield; | ||||
| const Scheduler = @import("scheduler.zig").Scheduler; | ||||
| const Logger = @import("../util.zig").Logger; | ||||
|  | ||||
| const File = std.fs.File; | ||||
| const log = std.log.scoped(.Arm7Tdmi); | ||||
|  | ||||
| // ARM Instructions | ||||
| pub const arm = struct { | ||||
|     pub const InstrFn = *const fn (*Arm7tdmi, *Bus, u32) void; | ||||
|     const lut: [0x1000]InstrFn = populate(); | ||||
|  | ||||
|     const processing = @import("cpu/arm/data_processing.zig").dataProcessing; | ||||
|     const psrTransfer = @import("cpu/arm/psr_transfer.zig").psrTransfer; | ||||
|     const transfer = @import("cpu/arm/single_data_transfer.zig").singleDataTransfer; | ||||
|     const halfSignedTransfer = @import("cpu/arm/half_signed_data_transfer.zig").halfAndSignedDataTransfer; | ||||
|     const blockTransfer = @import("cpu/arm/block_data_transfer.zig").blockDataTransfer; | ||||
|     const branch = @import("cpu/arm/branch.zig").branch; | ||||
|     const branchExchange = @import("cpu/arm/branch.zig").branchAndExchange; | ||||
|     const swi = @import("cpu/arm/software_interrupt.zig").armSoftwareInterrupt; | ||||
|     const swap = @import("cpu/arm/single_data_swap.zig").singleDataSwap; | ||||
|  | ||||
|     const multiply = @import("cpu/arm/multiply.zig").multiply; | ||||
|     const multiplyLong = @import("cpu/arm/multiply.zig").multiplyLong; | ||||
|  | ||||
|     /// Determine index into ARM InstrFn LUT | ||||
|     fn idx(opcode: u32) u12 { | ||||
|         return @truncate(u12, opcode >> 20 & 0xFF) << 4 | @truncate(u12, opcode >> 4 & 0xF); | ||||
|     } | ||||
|  | ||||
|     // Undefined ARM Instruction handler | ||||
|     fn und(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { | ||||
|         const id = idx(opcode); | ||||
|         cpu.panic("[CPU/Decode] ID: 0x{X:0>3} 0x{X:0>8} is an illegal opcode", .{ id, opcode }); | ||||
|     } | ||||
|  | ||||
|     fn populate() [0x1000]InstrFn { | ||||
|         return comptime { | ||||
|             @setEvalBranchQuota(0xE000); | ||||
|             var ret = [_]InstrFn{und} ** 0x1000; | ||||
|  | ||||
|             var i: usize = 0; | ||||
|             while (i < ret.len) : (i += 1) { | ||||
|                 ret[i] = switch (@as(u2, i >> 10)) { | ||||
|                     0b00 => if (i == 0x121) blk: { | ||||
|                         break :blk branchExchange; | ||||
|                     } else if (i & 0xFCF == 0x009) blk: { | ||||
|                         const A = i >> 5 & 1 == 1; | ||||
|                         const S = i >> 4 & 1 == 1; | ||||
|                         break :blk multiply(A, S); | ||||
|                     } else if (i & 0xFBF == 0x109) blk: { | ||||
|                         const B = i >> 6 & 1 == 1; | ||||
|                         break :blk swap(B); | ||||
|                     } else if (i & 0xF8F == 0x089) blk: { | ||||
|                         const U = i >> 6 & 1 == 1; | ||||
|                         const A = i >> 5 & 1 == 1; | ||||
|                         const S = i >> 4 & 1 == 1; | ||||
|                         break :blk multiplyLong(U, A, S); | ||||
|                     } else if (i & 0xE49 == 0x009 or i & 0xE49 == 0x049) blk: { | ||||
|                         const P = i >> 8 & 1 == 1; | ||||
|                         const U = i >> 7 & 1 == 1; | ||||
|                         const I = i >> 6 & 1 == 1; | ||||
|                         const W = i >> 5 & 1 == 1; | ||||
|                         const L = i >> 4 & 1 == 1; | ||||
|                         break :blk halfSignedTransfer(P, U, I, W, L); | ||||
|                     } else if (i & 0xD90 == 0x100) blk: { | ||||
|                         const I = i >> 9 & 1 == 1; | ||||
|                         const R = i >> 6 & 1 == 1; | ||||
|                         const kind = i >> 4 & 0x3; | ||||
|                         break :blk psrTransfer(I, R, kind); | ||||
|                     } else blk: { | ||||
|                         const I = i >> 9 & 1 == 1; | ||||
|                         const S = i >> 4 & 1 == 1; | ||||
|                         const instrKind = i >> 5 & 0xF; | ||||
|                         break :blk processing(I, S, instrKind); | ||||
|                     }, | ||||
|                     0b01 => if (i >> 9 & 1 == 1 and i & 1 == 1) und else blk: { | ||||
|                         const I = i >> 9 & 1 == 1; | ||||
|                         const P = i >> 8 & 1 == 1; | ||||
|                         const U = i >> 7 & 1 == 1; | ||||
|                         const B = i >> 6 & 1 == 1; | ||||
|                         const W = i >> 5 & 1 == 1; | ||||
|                         const L = i >> 4 & 1 == 1; | ||||
|                         break :blk transfer(I, P, U, B, W, L); | ||||
|                     }, | ||||
|                     else => switch (@as(u2, i >> 9 & 0x3)) { | ||||
|                         // MSB is guaranteed to be 1 | ||||
|                         0b00 => blk: { | ||||
|                             const P = i >> 8 & 1 == 1; | ||||
|                             const U = i >> 7 & 1 == 1; | ||||
|                             const S = i >> 6 & 1 == 1; | ||||
|                             const W = i >> 5 & 1 == 1; | ||||
|                             const L = i >> 4 & 1 == 1; | ||||
|                             break :blk blockTransfer(P, U, S, W, L); | ||||
|                         }, | ||||
|                         0b01 => blk: { | ||||
|                             const L = i >> 8 & 1 == 1; | ||||
|                             break :blk branch(L); | ||||
|                         }, | ||||
|                         0b10 => und, // COP Data Transfer | ||||
|                         0b11 => if (i >> 8 & 1 == 1) swi() else und, // COP Data Operation + Register Transfer | ||||
|                     }, | ||||
|                 }; | ||||
|             } | ||||
|  | ||||
|             return ret; | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| // THUMB Instructions | ||||
| pub const thumb = struct { | ||||
|     pub const InstrFn = *const fn (*Arm7tdmi, *Bus, u16) void; | ||||
|     const lut: [0x400]InstrFn = populate(); | ||||
|  | ||||
|     const processing = @import("cpu/thumb/data_processing.zig"); | ||||
|     const alu = @import("cpu/thumb/alu.zig").fmt4; | ||||
|     const transfer = @import("cpu/thumb/data_transfer.zig"); | ||||
|     const block_transfer = @import("cpu/thumb/block_data_transfer.zig"); | ||||
|     const swi = @import("cpu/thumb/software_interrupt.zig").fmt17; | ||||
|     const branch = @import("cpu/thumb/branch.zig"); | ||||
|  | ||||
|     /// Determine index into THUMB InstrFn LUT | ||||
|     fn idx(opcode: u16) u10 { | ||||
|         return @truncate(u10, opcode >> 6); | ||||
|     } | ||||
|  | ||||
|     /// Undefined THUMB Instruction Handler | ||||
|     fn und(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|         const id = idx(opcode); | ||||
|         cpu.panic("[CPU/Decode] ID: 0b{b:0>10} 0x{X:0>2} is an illegal opcode", .{ id, opcode }); | ||||
|     } | ||||
|  | ||||
|     fn populate() [0x400]InstrFn { | ||||
|         return comptime { | ||||
|             @setEvalBranchQuota(5025); // This is exact | ||||
|             var ret = [_]InstrFn{und} ** 0x400; | ||||
|  | ||||
|             var i: usize = 0; | ||||
|             while (i < ret.len) : (i += 1) { | ||||
|                 ret[i] = switch (@as(u3, i >> 7 & 0x7)) { | ||||
|                     0b000 => if (i >> 5 & 0x3 == 0b11) blk: { | ||||
|                         const I = i >> 4 & 1 == 1; | ||||
|                         const is_sub = i >> 3 & 1 == 1; | ||||
|                         const rn = i & 0x7; | ||||
|                         break :blk processing.fmt2(I, is_sub, rn); | ||||
|                     } else blk: { | ||||
|                         const op = i >> 5 & 0x3; | ||||
|                         const offset = i & 0x1F; | ||||
|                         break :blk processing.fmt1(op, offset); | ||||
|                     }, | ||||
|                     0b001 => blk: { | ||||
|                         const op = i >> 5 & 0x3; | ||||
|                         const rd = i >> 2 & 0x7; | ||||
|                         break :blk processing.fmt3(op, rd); | ||||
|                     }, | ||||
|                     0b010 => switch (@as(u2, i >> 5 & 0x3)) { | ||||
|                         0b00 => if (i >> 4 & 1 == 1) blk: { | ||||
|                             const op = i >> 2 & 0x3; | ||||
|                             const h1 = i >> 1 & 1; | ||||
|                             const h2 = i & 1; | ||||
|                             break :blk processing.fmt5(op, h1, h2); | ||||
|                         } else blk: { | ||||
|                             const op = i & 0xF; | ||||
|                             break :blk alu(op); | ||||
|                         }, | ||||
|                         0b01 => blk: { | ||||
|                             const rd = i >> 2 & 0x7; | ||||
|                             break :blk transfer.fmt6(rd); | ||||
|                         }, | ||||
|                         else => blk: { | ||||
|                             const op = i >> 4 & 0x3; | ||||
|                             const T = i >> 3 & 1 == 1; | ||||
|                             break :blk transfer.fmt78(op, T); | ||||
|                         }, | ||||
|                     }, | ||||
|                     0b011 => blk: { | ||||
|                         const B = i >> 6 & 1 == 1; | ||||
|                         const L = i >> 5 & 1 == 1; | ||||
|                         const offset = i & 0x1F; | ||||
|                         break :blk transfer.fmt9(B, L, offset); | ||||
|                     }, | ||||
|                     else => switch (@as(u3, i >> 6 & 0x7)) { | ||||
|                         // MSB is guaranteed to be 1 | ||||
|                         0b000 => blk: { | ||||
|                             const L = i >> 5 & 1 == 1; | ||||
|                             const offset = i & 0x1F; | ||||
|                             break :blk transfer.fmt10(L, offset); | ||||
|                         }, | ||||
|                         0b001 => blk: { | ||||
|                             const L = i >> 5 & 1 == 1; | ||||
|                             const rd = i >> 2 & 0x7; | ||||
|                             break :blk transfer.fmt11(L, rd); | ||||
|                         }, | ||||
|                         0b010 => blk: { | ||||
|                             const isSP = i >> 5 & 1 == 1; | ||||
|                             const rd = i >> 2 & 0x7; | ||||
|                             break :blk processing.fmt12(isSP, rd); | ||||
|                         }, | ||||
|                         0b011 => if (i >> 4 & 1 == 1) blk: { | ||||
|                             const L = i >> 5 & 1 == 1; | ||||
|                             const R = i >> 2 & 1 == 1; | ||||
|                             break :blk block_transfer.fmt14(L, R); | ||||
|                         } else blk: { | ||||
|                             const S = i >> 1 & 1 == 1; | ||||
|                             break :blk processing.fmt13(S); | ||||
|                         }, | ||||
|                         0b100 => blk: { | ||||
|                             const L = i >> 5 & 1 == 1; | ||||
|                             const rb = i >> 2 & 0x7; | ||||
|  | ||||
|                             break :blk block_transfer.fmt15(L, rb); | ||||
|                         }, | ||||
|                         0b101 => if (i >> 2 & 0xF == 0b1111) blk: { | ||||
|                             break :blk thumb.swi(); | ||||
|                         } else blk: { | ||||
|                             const cond = i >> 2 & 0xF; | ||||
|                             break :blk branch.fmt16(cond); | ||||
|                         }, | ||||
|                         0b110 => branch.fmt18(), | ||||
|                         0b111 => blk: { | ||||
|                             const is_low = i >> 5 & 1 == 1; | ||||
|                             break :blk branch.fmt19(is_low); | ||||
|                         }, | ||||
|                     }, | ||||
|                 }; | ||||
|             } | ||||
|  | ||||
|             return ret; | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| pub const Arm7tdmi = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     r: [16]u32, | ||||
|     pipe: Pipeline, | ||||
|     sched: *Scheduler, | ||||
|     bus: *Bus, | ||||
|     cpsr: PSR, | ||||
|     spsr: PSR, | ||||
|  | ||||
|     bank: Bank, | ||||
|  | ||||
|     logger: ?Logger, | ||||
|  | ||||
|     /// Bank of Registers from other CPU Modes | ||||
|     const Bank = struct { | ||||
|         /// Storage for r13_<mode>, r14_<mode> | ||||
|         /// e.g. [r13, r14, r13_svc, r14_svc] | ||||
|         r: [2 * 6]u32, | ||||
|  | ||||
|         /// Storage  for R8_fiq -> R12_fiq and their normal counterparts | ||||
|         /// e.g [r[0 + 8], fiq_r[0 + 8], r[1 + 8], fiq_r[1 + 8]...] | ||||
|         fiq: [2 * 5]u32, | ||||
|  | ||||
|         spsr: [5]PSR, | ||||
|  | ||||
|         const Kind = enum(u1) { | ||||
|             R13 = 0, | ||||
|             R14, | ||||
|         }; | ||||
|  | ||||
|         pub fn create() Bank { | ||||
|             return .{ | ||||
|                 .r = [_]u32{0x00} ** 12, | ||||
|                 .fiq = [_]u32{0x00} ** 10, | ||||
|                 .spsr = [_]PSR{.{ .raw = 0x0000_0000 }} ** 5, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         inline fn regIdx(mode: Mode, kind: Kind) usize { | ||||
|             const idx: usize = switch (mode) { | ||||
|                 .User, .System => 0, | ||||
|                 .Supervisor => 1, | ||||
|                 .Abort => 2, | ||||
|                 .Undefined => 3, | ||||
|                 .Irq => 4, | ||||
|                 .Fiq => 5, | ||||
|             }; | ||||
|  | ||||
|             return (idx * 2) + if (kind == .R14) @as(usize, 1) else 0; | ||||
|         } | ||||
|  | ||||
|         inline fn spsrIdx(mode: Mode) usize { | ||||
|             return switch (mode) { | ||||
|                 .Supervisor => 0, | ||||
|                 .Abort => 1, | ||||
|                 .Undefined => 2, | ||||
|                 .Irq => 3, | ||||
|                 .Fiq => 4, | ||||
|                 else => std.debug.panic("[CPU/Mode] {} does not have a SPSR Register", .{mode}), | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         inline fn fiqIdx(i: usize, mode: Mode) usize { | ||||
|             return (i * 2) + if (mode == .Fiq) @as(usize, 1) else 0; | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     pub fn init(sched: *Scheduler, bus: *Bus, log_file: ?std.fs.File) Self { | ||||
|         return Self{ | ||||
|             .r = [_]u32{0x00} ** 16, | ||||
|             .pipe = Pipeline.init(), | ||||
|             .sched = sched, | ||||
|             .bus = bus, | ||||
|             .cpsr = .{ .raw = 0x0000_001F }, | ||||
|             .spsr = .{ .raw = 0x0000_0000 }, | ||||
|             .bank = Bank.create(), | ||||
|             .logger = if (log_file) |file| Logger.init(file) else null, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub inline fn hasSPSR(self: *const Self) bool { | ||||
|         const mode = getModeChecked(self, self.cpsr.mode.read()); | ||||
|         return switch (mode) { | ||||
|             .System, .User => false, | ||||
|             else => true, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub inline fn isPrivileged(self: *const Self) bool { | ||||
|         const mode = getModeChecked(self, self.cpsr.mode.read()); | ||||
|         return switch (mode) { | ||||
|             .User => false, | ||||
|             else => true, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub inline fn isHalted(self: *const Self) bool { | ||||
|         return self.bus.io.haltcnt == .Halt; | ||||
|     } | ||||
|  | ||||
|     pub fn setCpsr(self: *Self, value: u32) void { | ||||
|         if (value & 0x1F != self.cpsr.raw & 0x1F) self.changeModeFromIdx(@truncate(u5, value & 0x1F)); | ||||
|         self.cpsr.raw = value; | ||||
|     } | ||||
|  | ||||
|     fn changeModeFromIdx(self: *Self, next: u5) void { | ||||
|         self.changeMode(getModeChecked(self, next)); | ||||
|     } | ||||
|  | ||||
|     pub fn setUserModeRegister(self: *Self, idx: usize, value: u32) void { | ||||
|         const current = getModeChecked(self, self.cpsr.mode.read()); | ||||
|  | ||||
|         switch (idx) { | ||||
|             8...12 => { | ||||
|                 if (current == .Fiq) { | ||||
|                     self.bank.fiq[Bank.fiqIdx(idx - 8, .User)] = value; | ||||
|                 } else self.r[idx] = value; | ||||
|             }, | ||||
|             13, 14 => switch (current) { | ||||
|                 .User, .System => self.r[idx] = value, | ||||
|                 else => { | ||||
|                     const kind = std.meta.intToEnum(Bank.Kind, idx - 13) catch unreachable; | ||||
|                     self.bank.r[Bank.regIdx(.User, kind)] = value; | ||||
|                 }, | ||||
|             }, | ||||
|             else => self.r[idx] = value, // R0 -> R7  and R15 | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn getUserModeRegister(self: *Self, idx: usize) u32 { | ||||
|         const current = getModeChecked(self, self.cpsr.mode.read()); | ||||
|  | ||||
|         return switch (idx) { | ||||
|             8...12 => if (current == .Fiq) self.bank.fiq[Bank.fiqIdx(idx - 8, .User)] else self.r[idx], | ||||
|             13, 14 => switch (current) { | ||||
|                 .User, .System => self.r[idx], | ||||
|                 else => blk: { | ||||
|                     const kind = std.meta.intToEnum(Bank.Kind, idx - 13) catch unreachable; | ||||
|                     break :blk self.bank.r[Bank.regIdx(.User, kind)]; | ||||
|                 }, | ||||
|             }, | ||||
|             else => self.r[idx], // R0 -> R7  and R15 | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn changeMode(self: *Self, next: Mode) void { | ||||
|         const now = getModeChecked(self, self.cpsr.mode.read()); | ||||
|  | ||||
|         // Bank R8 -> r12 | ||||
|         var i: usize = 0; | ||||
|         while (i < 5) : (i += 1) { | ||||
|             self.bank.fiq[Bank.fiqIdx(i, now)] = self.r[8 + i]; | ||||
|         } | ||||
|  | ||||
|         // Bank r13, r14, SPSR | ||||
|         switch (now) { | ||||
|             .User, .System => { | ||||
|                 self.bank.r[Bank.regIdx(now, .R13)] = self.r[13]; | ||||
|                 self.bank.r[Bank.regIdx(now, .R14)] = self.r[14]; | ||||
|             }, | ||||
|             else => { | ||||
|                 self.bank.r[Bank.regIdx(now, .R13)] = self.r[13]; | ||||
|                 self.bank.r[Bank.regIdx(now, .R14)] = self.r[14]; | ||||
|                 self.bank.spsr[Bank.spsrIdx(now)] = self.spsr; | ||||
|             }, | ||||
|         } | ||||
|  | ||||
|         // Grab R8 -> R12 | ||||
|         i = 0; | ||||
|         while (i < 5) : (i += 1) { | ||||
|             self.r[8 + i] = self.bank.fiq[Bank.fiqIdx(i, next)]; | ||||
|         } | ||||
|  | ||||
|         // Grab r13, r14, SPSR | ||||
|         switch (next) { | ||||
|             .User, .System => { | ||||
|                 self.r[13] = self.bank.r[Bank.regIdx(next, .R13)]; | ||||
|                 self.r[14] = self.bank.r[Bank.regIdx(next, .R14)]; | ||||
|             }, | ||||
|             else => { | ||||
|                 self.r[13] = self.bank.r[Bank.regIdx(next, .R13)]; | ||||
|                 self.r[14] = self.bank.r[Bank.regIdx(next, .R14)]; | ||||
|                 self.spsr = self.bank.spsr[Bank.spsrIdx(next)]; | ||||
|             }, | ||||
|         } | ||||
|  | ||||
|         self.cpsr.mode.write(@enumToInt(next)); | ||||
|     } | ||||
|  | ||||
|     /// Advances state so that the BIOS is skipped | ||||
|     /// | ||||
|     /// Note: This accesses the CPU's bus ptr so it only may be called | ||||
|     /// once the Bus has been properly initialized | ||||
|     /// | ||||
|     /// TODO: Make above notice impossible to do in code | ||||
|     pub fn fastBoot(self: *Self) void { | ||||
|         self.r = std.mem.zeroes([16]u32); | ||||
|  | ||||
|         // self.r[0] = 0x08000000; | ||||
|         // self.r[1] = 0x000000EA; | ||||
|         self.r[13] = 0x0300_7F00; | ||||
|         self.r[15] = 0x0800_0000; | ||||
|  | ||||
|         self.bank.r[Bank.regIdx(.Irq, .R13)] = 0x0300_7FA0; | ||||
|         self.bank.r[Bank.regIdx(.Supervisor, .R13)] = 0x0300_7FE0; | ||||
|  | ||||
|         // self.cpsr.raw = 0x6000001F; | ||||
|         self.cpsr.raw = 0x0000_001F; | ||||
|  | ||||
|         self.bus.bios.addr_latch = 0x0000_00DC + 8; | ||||
|     } | ||||
|  | ||||
|     pub fn step(self: *Self) void { | ||||
|         defer { | ||||
|             if (!self.pipe.flushed) self.r[15] += if (self.cpsr.t.read()) 2 else @as(u32, 4); | ||||
|             self.pipe.flushed = false; | ||||
|         } | ||||
|  | ||||
|         if (self.cpsr.t.read()) { | ||||
|             const opcode = @truncate(u16, self.pipe.step(self, u16) orelse return); | ||||
|             if (self.logger) |*trace| trace.mgbaLog(self, opcode); | ||||
|  | ||||
|             thumb.lut[thumb.idx(opcode)](self, self.bus, opcode); | ||||
|         } else { | ||||
|             const opcode = self.pipe.step(self, u32) orelse return; | ||||
|             if (self.logger) |*trace| trace.mgbaLog(self, opcode); | ||||
|  | ||||
|             if (checkCond(self.cpsr, @truncate(u4, opcode >> 28))) { | ||||
|                 arm.lut[arm.idx(opcode)](self, self.bus, opcode); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn stepDmaTransfer(self: *Self) bool { | ||||
|         comptime var i: usize = 0; | ||||
|         inline while (i < 4) : (i += 1) { | ||||
|             if (self.bus.dma[i].in_progress) { | ||||
|                 self.bus.dma[i].step(self); | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     pub fn handleInterrupt(self: *Self) void { | ||||
|         const should_handle = self.bus.io.ie.raw & self.bus.io.irq.raw; | ||||
|  | ||||
|         // Return if IME is disabled, CPSR I is set or there is nothing to handle | ||||
|         if (!self.bus.io.ime or self.cpsr.i.read() or should_handle == 0) return; | ||||
|  | ||||
|         // If Pipeline isn't full, we have a bug | ||||
|         std.debug.assert(self.pipe.isFull()); | ||||
|  | ||||
|         // log.debug("Handling Interrupt!", .{}); | ||||
|         self.bus.io.haltcnt = .Execute; | ||||
|  | ||||
|         // FIXME: This seems weird, but retAddr.gba suggests I need to make these changes | ||||
|         const ret_addr = self.r[15] - if (self.cpsr.t.read()) 0 else @as(u32, 4); | ||||
|         const new_spsr = self.cpsr.raw; | ||||
|  | ||||
|         self.changeMode(.Irq); | ||||
|         self.cpsr.t.write(false); | ||||
|         self.cpsr.i.write(true); | ||||
|  | ||||
|         self.r[14] = ret_addr; | ||||
|         self.spsr.raw = new_spsr; | ||||
|         self.r[15] = 0x0000_0018; | ||||
|         self.pipe.reload(self); | ||||
|     } | ||||
|  | ||||
|     inline fn fetch(self: *Self, comptime T: type, address: u32) T { | ||||
|         comptime std.debug.assert(T == u32 or T == u16); // Opcode may be 32-bit (ARM) or 16-bit (THUMB) | ||||
|  | ||||
|         // Bus.read will advance the scheduler. There are different timings for CPU fetches, | ||||
|         // so we want to undo what Bus.read will apply. We can do this by caching the current tick | ||||
|         // This is very dumb. | ||||
|         // | ||||
|         // FIXME: Please rework this | ||||
|         const tick_cache = self.sched.tick; | ||||
|         defer self.sched.tick = tick_cache + Bus.fetch_timings[@boolToInt(T == u32)][@truncate(u4, address >> 24)]; | ||||
|  | ||||
|         return self.bus.read(T, address); | ||||
|     } | ||||
|  | ||||
|     pub fn panic(self: *const Self, comptime format: []const u8, args: anytype) noreturn { | ||||
|         var i: usize = 0; | ||||
|         while (i < 16) : (i += 4) { | ||||
|             const i_1 = i + 1; | ||||
|             const i_2 = i + 2; | ||||
|             const i_3 = i + 3; | ||||
|             std.debug.print("R{}: 0x{X:0>8}\tR{}: 0x{X:0>8}\tR{}: 0x{X:0>8}\tR{}: 0x{X:0>8}\n", .{ i, self.r[i], i_1, self.r[i_1], i_2, self.r[i_2], i_3, self.r[i_3] }); | ||||
|         } | ||||
|         std.debug.print("cpsr: 0x{X:0>8} ", .{self.cpsr.raw}); | ||||
|         self.cpsr.toString(); | ||||
|  | ||||
|         std.debug.print("spsr: 0x{X:0>8} ", .{self.spsr.raw}); | ||||
|         self.spsr.toString(); | ||||
|  | ||||
|         std.debug.print("pipeline: {??X:0>8}\n", .{self.pipe.stage}); | ||||
|  | ||||
|         if (self.cpsr.t.read()) { | ||||
|             const opcode = self.bus.dbgRead(u16, self.r[15] - 4); | ||||
|             const id = thumb.idx(opcode); | ||||
|             std.debug.print("opcode: ID: 0x{b:0>10} 0x{X:0>4}\n", .{ id, opcode }); | ||||
|         } else { | ||||
|             const opcode = self.bus.dbgRead(u32, self.r[15] - 4); | ||||
|             const id = arm.idx(opcode); | ||||
|             std.debug.print("opcode: ID: 0x{X:0>3} 0x{X:0>8}\n", .{ id, opcode }); | ||||
|         } | ||||
|  | ||||
|         std.debug.print("tick: {}\n\n", .{self.sched.tick}); | ||||
|  | ||||
|         std.debug.panic(format, args); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const condition_lut = [_]u16{ | ||||
|     0xF0F0, // EQ - Equal | ||||
|     0x0F0F, // NE - Not Equal | ||||
|     0xCCCC, // CS - Unsigned higher or same | ||||
|     0x3333, // CC - Unsigned lower | ||||
|     0xFF00, // MI - Negative | ||||
|     0x00FF, // PL - Positive or Zero | ||||
|     0xAAAA, // VS - Overflow | ||||
|     0x5555, // VC - No Overflow | ||||
|     0x0C0C, // HI - unsigned hierh | ||||
|     0xF3F3, // LS - unsigned lower or same | ||||
|     0xAA55, // GE - greater or equal | ||||
|     0x55AA, // LT - less than | ||||
|     0x0A05, // GT - greater than | ||||
|     0xF5FA, // LE - less than or equal | ||||
|     0xFFFF, // AL - always | ||||
|     0x0000, // NV - never | ||||
| }; | ||||
|  | ||||
| pub inline fn checkCond(cpsr: PSR, cond: u4) bool { | ||||
|     const flags = @truncate(u4, cpsr.raw >> 28); | ||||
|  | ||||
|     return condition_lut[cond] & (@as(u16, 1) << flags) != 0; | ||||
| } | ||||
|  | ||||
| const Pipeline = struct { | ||||
|     const Self = @This(); | ||||
|     stage: [2]?u32, | ||||
|     flushed: bool, | ||||
|  | ||||
|     fn init() Self { | ||||
|         return .{ | ||||
|             .stage = [_]?u32{null} ** 2, | ||||
|             .flushed = false, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn isFull(self: *const Self) bool { | ||||
|         return self.stage[0] != null and self.stage[1] != null; | ||||
|     } | ||||
|  | ||||
|     pub fn step(self: *Self, cpu: *Arm7tdmi, comptime T: type) ?u32 { | ||||
|         comptime std.debug.assert(T == u32 or T == u16); | ||||
|  | ||||
|         const opcode = self.stage[0]; | ||||
|         self.stage[0] = self.stage[1]; | ||||
|         self.stage[1] = cpu.fetch(T, cpu.r[15]); | ||||
|  | ||||
|         return opcode; | ||||
|     } | ||||
|  | ||||
|     pub fn reload(self: *Self, cpu: *Arm7tdmi) void { | ||||
|         if (cpu.cpsr.t.read()) { | ||||
|             self.stage[0] = cpu.fetch(u16, cpu.r[15]); | ||||
|             self.stage[1] = cpu.fetch(u16, cpu.r[15] + 2); | ||||
|             cpu.r[15] += 4; | ||||
|         } else { | ||||
|             self.stage[0] = cpu.fetch(u32, cpu.r[15]); | ||||
|             self.stage[1] = cpu.fetch(u32, cpu.r[15] + 4); | ||||
|             cpu.r[15] += 8; | ||||
|         } | ||||
|  | ||||
|         self.flushed = true; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| pub const PSR = extern union { | ||||
|     mode: Bitfield(u32, 0, 5), | ||||
|     t: Bit(u32, 5), | ||||
|     f: Bit(u32, 6), | ||||
|     i: Bit(u32, 7), | ||||
|     v: Bit(u32, 28), | ||||
|     c: Bit(u32, 29), | ||||
|     z: Bit(u32, 30), | ||||
|     n: Bit(u32, 31), | ||||
|     raw: u32, | ||||
|  | ||||
|     fn toString(self: PSR) void { | ||||
|         std.debug.print("[", .{}); | ||||
|  | ||||
|         if (self.n.read()) std.debug.print("N", .{}) else std.debug.print("-", .{}); | ||||
|         if (self.z.read()) std.debug.print("Z", .{}) else std.debug.print("-", .{}); | ||||
|         if (self.c.read()) std.debug.print("C", .{}) else std.debug.print("-", .{}); | ||||
|         if (self.v.read()) std.debug.print("V", .{}) else std.debug.print("-", .{}); | ||||
|         if (self.i.read()) std.debug.print("I", .{}) else std.debug.print("-", .{}); | ||||
|         if (self.f.read()) std.debug.print("F", .{}) else std.debug.print("-", .{}); | ||||
|         if (self.t.read()) std.debug.print("T", .{}) else std.debug.print("-", .{}); | ||||
|         std.debug.print("|", .{}); | ||||
|         if (getMode(self.mode.read())) |m| std.debug.print("{s}", .{m.toString()}) else std.debug.print("---", .{}); | ||||
|  | ||||
|         std.debug.print("]\n", .{}); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Mode = enum(u5) { | ||||
|     User = 0b10000, | ||||
|     Fiq = 0b10001, | ||||
|     Irq = 0b10010, | ||||
|     Supervisor = 0b10011, | ||||
|     Abort = 0b10111, | ||||
|     Undefined = 0b11011, | ||||
|     System = 0b11111, | ||||
|  | ||||
|     fn toString(self: Mode) []const u8 { | ||||
|         return switch (self) { | ||||
|             .User => "usr", | ||||
|             .Fiq => "fiq", | ||||
|             .Irq => "irq", | ||||
|             .Supervisor => "svc", | ||||
|             .Abort => "abt", | ||||
|             .Undefined => "und", | ||||
|             .System => "sys", | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| fn getMode(bits: u5) ?Mode { | ||||
|     return std.meta.intToEnum(Mode, bits) catch null; | ||||
| } | ||||
|  | ||||
| fn getModeChecked(cpu: *const Arm7tdmi, bits: u5) Mode { | ||||
|     return getMode(bits) orelse cpu.panic("[CPU/CPSR] 0b{b:0>5} is an invalid CPU mode", .{bits}); | ||||
| } | ||||
| @@ -1,26 +0,0 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
|  | ||||
| const sext = @import("../../../util.zig").sext; | ||||
|  | ||||
| pub fn branch(comptime L: bool) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { | ||||
|             if (L) cpu.r[14] = cpu.r[15] - 4; | ||||
|  | ||||
|             cpu.r[15] +%= sext(u32, u24, opcode) << 2; | ||||
|             cpu.pipe.reload(cpu); | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn branchAndExchange(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { | ||||
|     const rn = opcode & 0xF; | ||||
|  | ||||
|     const thumb = cpu.r[rn] & 1 == 1; | ||||
|     cpu.r[15] = cpu.r[rn] & if (thumb) ~@as(u32, 1) else ~@as(u32, 3); | ||||
|  | ||||
|     cpu.cpsr.t.write(thumb); | ||||
|     cpu.pipe.reload(cpu); | ||||
| } | ||||
| @@ -1,183 +0,0 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
|  | ||||
| const exec = @import("../barrel_shifter.zig").exec; | ||||
| const ror = @import("../barrel_shifter.zig").ror; | ||||
|  | ||||
| pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime kind: u4) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { | ||||
|             const rd = @truncate(u4, opcode >> 12 & 0xF); | ||||
|             const rn = opcode >> 16 & 0xF; | ||||
|             const old_carry = @boolToInt(cpu.cpsr.c.read()); | ||||
|  | ||||
|             // If certain conditions are met, PC is 12 ahead instead of 8 | ||||
|             // TODO: Why these conditions? | ||||
|             if (!I and opcode >> 4 & 1 == 1) cpu.r[15] += 4; | ||||
|             const op1 = cpu.r[rn]; | ||||
|  | ||||
|             const amount = @truncate(u8, (opcode >> 8 & 0xF) << 1); | ||||
|             const op2 = if (I) ror(S, &cpu.cpsr, opcode & 0xFF, amount) else exec(S, cpu, opcode); | ||||
|  | ||||
|             // Undo special condition from above | ||||
|             if (!I and opcode >> 4 & 1 == 1) cpu.r[15] -= 4; | ||||
|  | ||||
|             var result: u32 = undefined; | ||||
|             var overflow: bool = undefined; | ||||
|  | ||||
|             // Perform Data Processing Logic | ||||
|             switch (kind) { | ||||
|                 0x0 => result = op1 & op2, // AND | ||||
|                 0x1 => result = op1 ^ op2, // EOR | ||||
|                 0x2 => result = op1 -% op2, // SUB | ||||
|                 0x3 => result = op2 -% op1, // RSB | ||||
|                 0x4 => result = add(&overflow, op1, op2), // ADD | ||||
|                 0x5 => result = adc(&overflow, op1, op2, old_carry), // ADC | ||||
|                 0x6 => result = sbc(op1, op2, old_carry), // SBC | ||||
|                 0x7 => result = sbc(op2, op1, old_carry), // RSC | ||||
|                 0x8 => { | ||||
|                     // TST | ||||
|                     if (rd == 0xF) | ||||
|                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
|                     result = op1 & op2; | ||||
|                 }, | ||||
|                 0x9 => { | ||||
|                     // TEQ | ||||
|                     if (rd == 0xF) | ||||
|                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
|                     result = op1 ^ op2; | ||||
|                 }, | ||||
|                 0xA => { | ||||
|                     // CMP | ||||
|                     if (rd == 0xF) | ||||
|                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
|                     result = op1 -% op2; | ||||
|                 }, | ||||
|                 0xB => { | ||||
|                     // CMN | ||||
|                     if (rd == 0xF) | ||||
|                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
|                     overflow = @addWithOverflow(u32, op1, op2, &result); | ||||
|                 }, | ||||
|                 0xC => result = op1 | op2, // ORR | ||||
|                 0xD => result = op2, // MOV | ||||
|                 0xE => result = op1 & ~op2, // BIC | ||||
|                 0xF => result = ~op2, // MVN | ||||
|             } | ||||
|  | ||||
|             // Write to Destination Register | ||||
|             switch (kind) { | ||||
|                 0x8, 0x9, 0xA, 0xB => {}, // Test Operations | ||||
|                 else => { | ||||
|                     cpu.r[rd] = result; | ||||
|                     if (rd == 0xF) { | ||||
|                         if (S) cpu.setCpsr(cpu.spsr.raw); | ||||
|                         cpu.pipe.reload(cpu); | ||||
|                     } | ||||
|                 }, | ||||
|             } | ||||
|  | ||||
|             // Write Flags | ||||
|             switch (kind) { | ||||
|                 0x0, 0x1, 0xC, 0xD, 0xE, 0xF => if (S and rd != 0xF) { | ||||
|                     // Logic Operation Flags | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|                     // C set by Barrel Shifter, V is unaffected | ||||
|  | ||||
|                 }, | ||||
|                 0x2, 0x3 => if (S and rd != 0xF) { | ||||
|                     // SUB, RSB Flags | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|  | ||||
|                     if (kind == 0x2) { | ||||
|                         // SUB specific | ||||
|                         cpu.cpsr.c.write(op2 <= op1); | ||||
|                         cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                     } else { | ||||
|                         // RSB Specific | ||||
|                         cpu.cpsr.c.write(op1 <= op2); | ||||
|                         cpu.cpsr.v.write(((op2 ^ result) & (~op1 ^ result)) >> 31 & 1 == 1); | ||||
|                     } | ||||
|                 }, | ||||
|                 0x4, 0x5 => if (S and rd != 0xF) { | ||||
|                     // ADD, ADC Flags | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|                     cpu.cpsr.c.write(overflow); | ||||
|                     cpu.cpsr.v.write(((op1 ^ result) & (op2 ^ result)) >> 31 & 1 == 1); | ||||
|                 }, | ||||
|                 0x6, 0x7 => if (S and rd != 0xF) { | ||||
|                     // SBC, RSC Flags | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|  | ||||
|                     if (kind == 0x6) { | ||||
|                         // SBC specific | ||||
|                         const subtrahend = @as(u64, op2) -% old_carry +% 1; | ||||
|                         cpu.cpsr.c.write(subtrahend <= op1); | ||||
|                         cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                     } else { | ||||
|                         // RSC Specific | ||||
|                         const subtrahend = @as(u64, op1) -% old_carry +% 1; | ||||
|                         cpu.cpsr.c.write(subtrahend <= op2); | ||||
|                         cpu.cpsr.v.write(((op2 ^ result) & (~op1 ^ result)) >> 31 & 1 == 1); | ||||
|                     } | ||||
|                 }, | ||||
|                 0x8, 0x9, 0xA, 0xB => { | ||||
|                     // Test Operation Flags | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|  | ||||
|                     if (kind == 0xA) { | ||||
|                         // CMP specific | ||||
|                         cpu.cpsr.c.write(op2 <= op1); | ||||
|                         cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                     } else if (kind == 0xB) { | ||||
|                         // CMN specific | ||||
|                         cpu.cpsr.c.write(overflow); | ||||
|                         cpu.cpsr.v.write(((op1 ^ result) & (op2 ^ result)) >> 31 & 1 == 1); | ||||
|                     } else { | ||||
|                         // TST, TEQ specific | ||||
|                         // Barrel Shifter should always calc CPSR C in TST | ||||
|                         if (!S) _ = exec(true, cpu, opcode); | ||||
|                     } | ||||
|                 }, | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn sbc(left: u32, right: u32, old_carry: u1) u32 { | ||||
|     // TODO: Make your own version (thanks peach.bot) | ||||
|     const subtrahend = @as(u64, right) -% old_carry +% 1; | ||||
|     const ret = @truncate(u32, left -% subtrahend); | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| pub fn add(overflow: *bool, left: u32, right: u32) u32 { | ||||
|     var ret: u32 = undefined; | ||||
|     overflow.* = @addWithOverflow(u32, left, right, &ret); | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| pub fn adc(overflow: *bool, left: u32, right: u32, old_carry: u1) u32 { | ||||
|     var ret: u32 = undefined; | ||||
|     const first = @addWithOverflow(u32, left, right, &ret); | ||||
|     const second = @addWithOverflow(u32, ret, old_carry, &ret); | ||||
|  | ||||
|     overflow.* = first or second; | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| fn undefinedTestBehaviour(cpu: *Arm7tdmi) void { | ||||
|     @setCold(true); | ||||
|     cpu.setCpsr(cpu.spsr.raw); | ||||
| } | ||||
| @@ -1,102 +0,0 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").thumb.InstrFn; | ||||
|  | ||||
| const adc = @import("../arm/data_processing.zig").adc; | ||||
| const sbc = @import("../arm/data_processing.zig").sbc; | ||||
|  | ||||
| const lsl = @import("../barrel_shifter.zig").lsl; | ||||
| const lsr = @import("../barrel_shifter.zig").lsr; | ||||
| const asr = @import("../barrel_shifter.zig").asr; | ||||
| const ror = @import("../barrel_shifter.zig").ror; | ||||
|  | ||||
| pub fn fmt4(comptime op: u4) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const rs = opcode >> 3 & 0x7; | ||||
|             const rd = opcode & 0x7; | ||||
|             const carry = @boolToInt(cpu.cpsr.c.read()); | ||||
|  | ||||
|             const op1 = cpu.r[rd]; | ||||
|             const op2 = cpu.r[rs]; | ||||
|  | ||||
|             var result: u32 = undefined; | ||||
|             var overflow: bool = undefined; | ||||
|             switch (op) { | ||||
|                 0x0 => result = op1 & op2, // AND | ||||
|                 0x1 => result = op1 ^ op2, // EOR | ||||
|                 0x2 => result = lsl(true, &cpu.cpsr, op1, @truncate(u8, op2)), // LSL | ||||
|                 0x3 => result = lsr(true, &cpu.cpsr, op1, @truncate(u8, op2)), // LSR | ||||
|                 0x4 => result = asr(true, &cpu.cpsr, op1, @truncate(u8, op2)), // ASR | ||||
|                 0x5 => result = adc(&overflow, op1, op2, carry), // ADC | ||||
|                 0x6 => result = sbc(op1, op2, carry), // SBC | ||||
|                 0x7 => result = ror(true, &cpu.cpsr, op1, @truncate(u8, op2)), // ROR | ||||
|                 0x8 => result = op1 & op2, // TST | ||||
|                 0x9 => result = 0 -% op2, // NEG | ||||
|                 0xA => result = op1 -% op2, // CMP | ||||
|                 0xB => overflow = @addWithOverflow(u32, op1, op2, &result), // CMN | ||||
|                 0xC => result = op1 | op2, // ORR | ||||
|                 0xD => result = @truncate(u32, @as(u64, op2) * @as(u64, op1)), | ||||
|                 0xE => result = op1 & ~op2, | ||||
|                 0xF => result = ~op2, | ||||
|             } | ||||
|  | ||||
|             // Write to Destination Register | ||||
|             switch (op) { | ||||
|                 0x8, 0xA, 0xB => {}, | ||||
|                 else => cpu.r[rd] = result, | ||||
|             } | ||||
|  | ||||
|             // Write Flags | ||||
|             switch (op) { | ||||
|                 0x0, 0x1, 0x2, 0x3, 0x4, 0x7, 0xC, 0xE, 0xF => { | ||||
|                     // Logic Operations | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|                     // C set by Barrel Shifter, V is unaffected | ||||
|                 }, | ||||
|                 0x8, 0xA => { | ||||
|                     // Test Flags | ||||
|                     // CMN (0xB) is handled with ADC | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|  | ||||
|                     if (op == 0xA) { | ||||
|                         // CMP specific | ||||
|                         cpu.cpsr.c.write(op2 <= op1); | ||||
|                         cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                     } | ||||
|                 }, | ||||
|                 0x5, 0xB => { | ||||
|                     // ADC, CMN | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|                     cpu.cpsr.c.write(overflow); | ||||
|                     cpu.cpsr.v.write(((op1 ^ result) & (op2 ^ result)) >> 31 & 1 == 1); | ||||
|                 }, | ||||
|                 0x6 => { | ||||
|                     // SBC | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|  | ||||
|                     const subtrahend = @as(u64, op2) -% carry +% 1; | ||||
|                     cpu.cpsr.c.write(subtrahend <= op1); | ||||
|                     cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                 }, | ||||
|                 0x9 => { | ||||
|                     // NEG | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|                     cpu.cpsr.c.write(op2 <= 0); | ||||
|                     cpu.cpsr.v.write(((0 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                 }, | ||||
|                 0xD => { | ||||
|                     // Multiplication | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|                     // V is unaffected, assuming similar behaviour to ARMv4 MUL C is undefined | ||||
|                 }, | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
| @@ -1,54 +0,0 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").thumb.InstrFn; | ||||
|  | ||||
| const checkCond = @import("../../cpu.zig").checkCond; | ||||
| const sext = @import("../../../util.zig").sext; | ||||
|  | ||||
| pub fn fmt16(comptime cond: u4) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             // B | ||||
|             if (cond == 0xE or cond == 0xF) | ||||
|                 cpu.panic("[CPU/THUMB.16] Undefined conditional branch with condition {}", .{cond}); | ||||
|  | ||||
|             if (!checkCond(cpu.cpsr, cond)) return; | ||||
|  | ||||
|             cpu.r[15] +%= sext(u32, u8, opcode & 0xFF) << 1; | ||||
|             cpu.pipe.reload(cpu); | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn fmt18() InstrFn { | ||||
|     return struct { | ||||
|         // B but conditional | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             cpu.r[15] +%= sext(u32, u11, opcode & 0x7FF) << 1; | ||||
|             cpu.pipe.reload(cpu); | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn fmt19(comptime is_low: bool) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             // BL | ||||
|             const offset = opcode & 0x7FF; | ||||
|  | ||||
|             if (is_low) { | ||||
|                 // Instruction 2 | ||||
|                 const next_opcode = cpu.r[15] - 2; | ||||
|  | ||||
|                 cpu.r[15] = cpu.r[14] +% (offset << 1); | ||||
|                 cpu.r[14] = next_opcode | 1; | ||||
|  | ||||
|                 cpu.pipe.reload(cpu); | ||||
|             } else { | ||||
|                 // Instruction 1 | ||||
|                 const lr_offset = sext(u32, u11, offset) << 12; | ||||
|                 cpu.r[14] = (cpu.r[15] +% lr_offset) & ~@as(u32, 1); | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
| @@ -1,199 +0,0 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").thumb.InstrFn; | ||||
|  | ||||
| const add = @import("../arm/data_processing.zig").add; | ||||
|  | ||||
| const lsl = @import("../barrel_shifter.zig").lsl; | ||||
| const lsr = @import("../barrel_shifter.zig").lsr; | ||||
| const asr = @import("../barrel_shifter.zig").asr; | ||||
|  | ||||
| pub fn fmt1(comptime op: u2, comptime offset: u5) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const rs = opcode >> 3 & 0x7; | ||||
|             const rd = opcode & 0x7; | ||||
|  | ||||
|             const result = switch (op) { | ||||
|                 0b00 => blk: { | ||||
|                     // LSL | ||||
|                     if (offset == 0) { | ||||
|                         break :blk cpu.r[rs]; | ||||
|                     } else { | ||||
|                         break :blk lsl(true, &cpu.cpsr, cpu.r[rs], offset); | ||||
|                     } | ||||
|                 }, | ||||
|                 0b01 => blk: { | ||||
|                     // LSR | ||||
|                     if (offset == 0) { | ||||
|                         cpu.cpsr.c.write(cpu.r[rs] >> 31 & 1 == 1); | ||||
|                         break :blk @as(u32, 0); | ||||
|                     } else { | ||||
|                         break :blk lsr(true, &cpu.cpsr, cpu.r[rs], offset); | ||||
|                     } | ||||
|                 }, | ||||
|                 0b10 => blk: { | ||||
|                     // ASR | ||||
|                     if (offset == 0) { | ||||
|                         cpu.cpsr.c.write(cpu.r[rs] >> 31 & 1 == 1); | ||||
|                         break :blk @bitCast(u32, @bitCast(i32, cpu.r[rs]) >> 31); | ||||
|                     } else { | ||||
|                         break :blk asr(true, &cpu.cpsr, cpu.r[rs], offset); | ||||
|                     } | ||||
|                 }, | ||||
|                 else => cpu.panic("[CPU/THUMB.1] 0b{b:0>2} is not a valid op", .{op}), | ||||
|             }; | ||||
|  | ||||
|             // Equivalent to an ARM MOVS | ||||
|             cpu.r[rd] = result; | ||||
|  | ||||
|             // Write Flags | ||||
|             cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|             cpu.cpsr.z.write(result == 0); | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn fmt5(comptime op: u2, comptime h1: u1, comptime h2: u1) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const rs = @as(u4, h2) << 3 | (opcode >> 3 & 0x7); | ||||
|             const rd = @as(u4, h1) << 3 | (opcode & 0x7); | ||||
|  | ||||
|             const op1 = cpu.r[rd]; | ||||
|             const op2 = cpu.r[rs]; | ||||
|  | ||||
|             var result: u32 = undefined; | ||||
|             var overflow: bool = undefined; | ||||
|             switch (op) { | ||||
|                 0b00 => result = add(&overflow, op1, op2), // ADD | ||||
|                 0b01 => result = op1 -% op2, // CMP | ||||
|                 0b10 => result = op2, // MOV | ||||
|                 0b11 => {}, | ||||
|             } | ||||
|  | ||||
|             // Write to Destination Register | ||||
|             switch (op) { | ||||
|                 0b01 => {}, // Test Instruction | ||||
|                 0b11 => { | ||||
|                     // BX | ||||
|                     const is_thumb = op2 & 1 == 1; | ||||
|                     cpu.r[15] = op2 & ~@as(u32, 1); | ||||
|  | ||||
|                     cpu.cpsr.t.write(is_thumb); | ||||
|                     cpu.pipe.reload(cpu); | ||||
|                 }, | ||||
|                 else => { | ||||
|                     cpu.r[rd] = result; | ||||
|                     if (rd == 0xF) { | ||||
|                         cpu.r[15] &= ~@as(u32, 1); | ||||
|                         cpu.pipe.reload(cpu); | ||||
|                     } | ||||
|                 }, | ||||
|             } | ||||
|  | ||||
|             // Write Flags | ||||
|             switch (op) { | ||||
|                 0b01 => { | ||||
|                     // CMP | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|                     cpu.cpsr.c.write(op2 <= op1); | ||||
|                     cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                 }, | ||||
|                 0b00, 0b10, 0b11 => {}, // MOV and Branch Instruction | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn fmt2(comptime I: bool, is_sub: bool, rn: u3) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const rs = opcode >> 3 & 0x7; | ||||
|             const rd = @truncate(u3, opcode); | ||||
|             const op1 = cpu.r[rs]; | ||||
|             const op2: u32 = if (I) rn else cpu.r[rn]; | ||||
|  | ||||
|             if (is_sub) { | ||||
|                 // SUB | ||||
|                 const result = op1 -% op2; | ||||
|                 cpu.r[rd] = result; | ||||
|  | ||||
|                 cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                 cpu.cpsr.z.write(result == 0); | ||||
|                 cpu.cpsr.c.write(op2 <= op1); | ||||
|                 cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|             } else { | ||||
|                 // ADD | ||||
|                 var overflow: bool = undefined; | ||||
|                 const result = add(&overflow, op1, op2); | ||||
|                 cpu.r[rd] = result; | ||||
|  | ||||
|                 cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                 cpu.cpsr.z.write(result == 0); | ||||
|                 cpu.cpsr.c.write(overflow); | ||||
|                 cpu.cpsr.v.write(((op1 ^ result) & (op2 ^ result)) >> 31 & 1 == 1); | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn fmt3(comptime op: u2, comptime rd: u3) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const op1 = cpu.r[rd]; | ||||
|             const op2: u32 = opcode & 0xFF; // Offset | ||||
|  | ||||
|             var overflow: bool = undefined; | ||||
|             const result: u32 = switch (op) { | ||||
|                 0b00 => op2, // MOV | ||||
|                 0b01 => op1 -% op2, // CMP | ||||
|                 0b10 => add(&overflow, op1, op2), // ADD | ||||
|                 0b11 => op1 -% op2, // SUB | ||||
|             }; | ||||
|  | ||||
|             // Write to Register | ||||
|             if (op != 0b01) cpu.r[rd] = result; | ||||
|  | ||||
|             // Write Flags | ||||
|             cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|             cpu.cpsr.z.write(result == 0); | ||||
|  | ||||
|             switch (op) { | ||||
|                 0b00 => {}, // MOV | C set by Barrel Shifter, V is unaffected | ||||
|                 0b01, 0b11 => { | ||||
|                     // SUB, CMP | ||||
|                     cpu.cpsr.c.write(op2 <= op1); | ||||
|                     cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                 }, | ||||
|                 0b10 => { | ||||
|                     // ADD | ||||
|                     cpu.cpsr.c.write(overflow); | ||||
|                     cpu.cpsr.v.write(((op1 ^ result) & (op2 ^ result)) >> 31 & 1 == 1); | ||||
|                 }, | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn fmt12(comptime isSP: bool, comptime rd: u3) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             // ADD | ||||
|             const left = if (isSP) cpu.r[13] else cpu.r[15] & ~@as(u32, 2); | ||||
|             const right = (opcode & 0xFF) << 2; | ||||
|             cpu.r[rd] = left + right; | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn fmt13(comptime S: bool) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             // ADD | ||||
|             const offset = (opcode & 0x7F) << 2; | ||||
|             cpu.r[13] = if (S) cpu.r[13] - offset else cpu.r[13] + offset; | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
							
								
								
									
										157
									
								
								src/core/emu.zig
									
									
									
									
									
								
							
							
						
						
									
										157
									
								
								src/core/emu.zig
									
									
									
									
									
								
							| @@ -1,157 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const SDL = @import("sdl2"); | ||||
| const config = @import("../config.zig"); | ||||
|  | ||||
| const Scheduler = @import("scheduler.zig").Scheduler; | ||||
| const Arm7tdmi = @import("cpu.zig").Arm7tdmi; | ||||
| const FpsTracker = @import("../util.zig").FpsTracker; | ||||
| const RingBuffer = @import("../util.zig").RingBuffer; | ||||
|  | ||||
| const Timer = std.time.Timer; | ||||
| const Atomic = std.atomic.Atomic; | ||||
|  | ||||
| /// 4 Cycles in 1 dot | ||||
| const cycles_per_dot = 4; | ||||
|  | ||||
| /// The GBA draws 228 Horizontal which each consist 308 dots | ||||
| /// (note: not all lines are visible) | ||||
| const cycles_per_frame = 228 * (308 * cycles_per_dot); //280896 | ||||
|  | ||||
| /// The GBA ARM7TDMI runs at 2^24 Hz | ||||
| const clock_rate = 1 << 24; // 16.78MHz | ||||
|  | ||||
| /// The # of nanoseconds a frame should take | ||||
| const frame_period = (std.time.ns_per_s * cycles_per_frame) / clock_rate; | ||||
|  | ||||
| /// Exact Value:  59.7275005696Hz | ||||
| /// The inverse of the frame period | ||||
| pub const frame_rate: f64 = @intToFloat(f64, clock_rate) / cycles_per_frame; | ||||
|  | ||||
| const log = std.log.scoped(.Emulation); | ||||
|  | ||||
| const RunKind = enum { | ||||
|     Unlimited, | ||||
|     UnlimitedFPS, | ||||
|     Limited, | ||||
|     LimitedFPS, | ||||
| }; | ||||
|  | ||||
| pub fn run(quit: *Atomic(bool), scheduler: *Scheduler, cpu: *Arm7tdmi, tracker: *FpsTracker) void { | ||||
|     const audio_sync = config.config().guest.audio_sync and !config.config().host.mute; | ||||
|     if (audio_sync) log.info("Audio sync enabled", .{}); | ||||
|  | ||||
|     if (config.config().guest.video_sync) { | ||||
|         inner(.LimitedFPS, audio_sync, quit, scheduler, cpu, tracker); | ||||
|     } else { | ||||
|         inner(.UnlimitedFPS, audio_sync, quit, scheduler, cpu, tracker); | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn inner(comptime kind: RunKind, audio_sync: bool, quit: *Atomic(bool), scheduler: *Scheduler, cpu: *Arm7tdmi, tracker: ?*FpsTracker) void { | ||||
|     if (kind == .UnlimitedFPS or kind == .LimitedFPS) { | ||||
|         std.debug.assert(tracker != null); | ||||
|         log.info("FPS tracking enabled", .{}); | ||||
|     } | ||||
|  | ||||
|     switch (kind) { | ||||
|         .Unlimited, .UnlimitedFPS => { | ||||
|             log.info("Emulation w/out video sync", .{}); | ||||
|  | ||||
|             while (!quit.load(.Monotonic)) { | ||||
|                 runFrame(scheduler, cpu); | ||||
|                 audioSync(audio_sync, &cpu.bus.apu.sample_queue); | ||||
|  | ||||
|                 if (kind == .UnlimitedFPS) tracker.?.tick(); | ||||
|             } | ||||
|         }, | ||||
|         .Limited, .LimitedFPS => { | ||||
|             log.info("Emulation w/ video sync", .{}); | ||||
|             var timer = Timer.start() catch @panic("failed to initalize std.timer.Timer"); | ||||
|             var wake_time: u64 = frame_period; | ||||
|  | ||||
|             while (!quit.load(.Monotonic)) { | ||||
|                 runFrame(scheduler, cpu); | ||||
|                 const new_wake_time = videoSync(&timer, wake_time); | ||||
|  | ||||
|                 // Spin to make up the difference of OS scheduler innacuracies | ||||
|                 // If we happen to also be syncing to audio, we choose to spin on | ||||
|                 // the amount of time needed for audio to catch up rather than | ||||
|                 // our expected wake-up time | ||||
|  | ||||
|                 audioSync(audio_sync, &cpu.bus.apu.sample_queue); | ||||
|                 if (!audio_sync) spinLoop(&timer, wake_time); | ||||
|                 wake_time = new_wake_time; | ||||
|  | ||||
|                 if (kind == .LimitedFPS) tracker.?.tick(); | ||||
|             } | ||||
|         }, | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn runFrame(sched: *Scheduler, cpu: *Arm7tdmi) void { | ||||
|     const frame_end = sched.tick + cycles_per_frame; | ||||
|  | ||||
|     while (sched.tick < frame_end) { | ||||
|         if (!cpu.stepDmaTransfer()) { | ||||
|             if (cpu.isHalted()) { | ||||
|                 // Fast-forward to next Event | ||||
|                 sched.tick = sched.queue.peek().?.tick; | ||||
|             } else { | ||||
|                 cpu.step(); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (sched.tick >= sched.nextTimestamp()) sched.handleEvent(cpu); | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn audioSync(audio_sync: bool, sample_queue: *RingBuffer(u16)) void { | ||||
|     comptime std.debug.assert(@import("../platform.zig").sample_format == SDL.AUDIO_U16); | ||||
|     // const sample_size = 2 * @sizeOf(u16); | ||||
|     // const max_buf_size: c_int = 0x400; | ||||
|  | ||||
|     _ = audio_sync; | ||||
|     _ = sample_queue; | ||||
| } | ||||
|  | ||||
| fn videoSync(timer: *Timer, wake_time: u64) u64 { | ||||
|     // Use the OS scheduler to put the emulation thread to sleep | ||||
|     const recalculated = sleep(timer, wake_time); | ||||
|  | ||||
|     // If sleep() determined we need to adjust our wake up time, do so | ||||
|     // otherwise predict our next wake up time according to the frame period | ||||
|     return recalculated orelse wake_time + frame_period; | ||||
| } | ||||
|  | ||||
| // TODO: Better sleep impl? | ||||
| fn sleep(timer: *Timer, wake_time: u64) ?u64 { | ||||
|     const timestamp = timer.read(); | ||||
|  | ||||
|     // ns_late is non zero if we are late. | ||||
|     var ns_late = timestamp -| wake_time; | ||||
|  | ||||
|     // If we're more than a frame late, skip the rest of this loop | ||||
|     // Recalculate what our new wake time should be so that we can | ||||
|     // get "back on track" | ||||
|     if (ns_late > frame_period) return timestamp + frame_period; | ||||
|     const sleep_for = frame_period - ns_late; | ||||
|  | ||||
|     const step = 2 * std.time.ns_per_ms; // Granularity of 2ms | ||||
|     const times = sleep_for / step; | ||||
|     var i: usize = 0; | ||||
|  | ||||
|     while (i < times) : (i += 1) { | ||||
|         std.time.sleep(step); | ||||
|  | ||||
|         // Upon wakeup, check to see if this particular sleep was longer than expected | ||||
|         // if so we should exit early, but probably not skip a whole frame period | ||||
|         ns_late = timer.read() -| wake_time; | ||||
|         if (ns_late > frame_period) return null; | ||||
|     } | ||||
|  | ||||
|     return null; | ||||
| } | ||||
|  | ||||
| fn spinLoop(timer: *Timer, wake_time: u64) void { | ||||
|     while (true) if (timer.read() > wake_time) break; | ||||
| } | ||||
							
								
								
									
										1506
									
								
								src/core/ppu.zig
									
									
									
									
									
								
							
							
						
						
									
										1506
									
								
								src/core/ppu.zig
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										685
									
								
								src/cpu.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										685
									
								
								src/cpu.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,685 @@ | ||||
| const std = @import("std"); | ||||
| const util = @import("util.zig"); | ||||
|  | ||||
| const Bus = @import("Bus.zig"); | ||||
| const Bit = @import("bitfield").Bit; | ||||
| const Bitfield = @import("bitfield").Bitfield; | ||||
| const Scheduler = @import("scheduler.zig").Scheduler; | ||||
| const FilePaths = @import("util.zig").FilePaths; | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
| const File = std.fs.File; | ||||
|  | ||||
| // ARM Instruction Groups | ||||
| const dataProcessing = @import("cpu/arm/data_processing.zig").dataProcessing; | ||||
| const psrTransfer = @import("cpu/arm/psr_transfer.zig").psrTransfer; | ||||
| const singleDataTransfer = @import("cpu/arm/single_data_transfer.zig").singleDataTransfer; | ||||
| const halfAndSignedDataTransfer = @import("cpu/arm/half_signed_data_transfer.zig").halfAndSignedDataTransfer; | ||||
| const blockDataTransfer = @import("cpu/arm/block_data_transfer.zig").blockDataTransfer; | ||||
| const branch = @import("cpu/arm/branch.zig").branch; | ||||
| const branchAndExchange = @import("cpu/arm/branch.zig").branchAndExchange; | ||||
| const armSoftwareInterrupt = @import("cpu/arm/software_interrupt.zig").armSoftwareInterrupt; | ||||
| const singleDataSwap = @import("cpu/arm/single_data_swap.zig").singleDataSwap; | ||||
|  | ||||
| const multiply = @import("cpu/arm/multiply.zig").multiply; | ||||
| const multiplyLong = @import("cpu/arm/multiply.zig").multiplyLong; | ||||
|  | ||||
| // THUMB Instruction Groups | ||||
| const format1 = @import("cpu/thumb/data_processing.zig").format1; | ||||
| const format2 = @import("cpu/thumb/data_processing.zig").format2; | ||||
| const format3 = @import("cpu/thumb/data_processing.zig").format3; | ||||
| const format12 = @import("cpu/thumb/data_processing.zig").format12; | ||||
| const format13 = @import("cpu/thumb/data_processing.zig").format13; | ||||
|  | ||||
| const format4 = @import("cpu/thumb/alu.zig").format4; | ||||
| const format5 = @import("cpu/thumb/processing_branch.zig").format5; | ||||
|  | ||||
| const format6 = @import("cpu/thumb/data_transfer.zig").format6; | ||||
| const format78 = @import("cpu/thumb/data_transfer.zig").format78; | ||||
| const format9 = @import("cpu/thumb/data_transfer.zig").format9; | ||||
| const format10 = @import("cpu/thumb/data_transfer.zig").format10; | ||||
| const format11 = @import("cpu/thumb/data_transfer.zig").format11; | ||||
| const format14 = @import("cpu/thumb/block_data_transfer.zig").format14; | ||||
| const format15 = @import("cpu/thumb/block_data_transfer.zig").format15; | ||||
|  | ||||
| const format16 = @import("cpu/thumb/branch.zig").format16; | ||||
| const format18 = @import("cpu/thumb/branch.zig").format18; | ||||
| const format19 = @import("cpu/thumb/branch.zig").format19; | ||||
|  | ||||
| const thumbSoftwareInterrupt = @import("cpu/thumb/software_interrupt.zig").thumbSoftwareInterrupt; | ||||
|  | ||||
| pub const ArmInstrFn = fn (*Arm7tdmi, *Bus, u32) void; | ||||
| pub const ThumbInstrFn = fn (*Arm7tdmi, *Bus, u16) void; | ||||
| const arm_lut: [0x1000]ArmInstrFn = armPopulate(); | ||||
| const thumb_lut: [0x400]ThumbInstrFn = thumbPopulate(); | ||||
|  | ||||
| const enable_logging = @import("main.zig").enable_logging; | ||||
| const log = std.log.scoped(.Arm7Tdmi); | ||||
|  | ||||
| pub const Arm7tdmi = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     r: [16]u32, | ||||
|     sched: *Scheduler, | ||||
|     bus: Bus, | ||||
|     cpsr: PSR, | ||||
|     spsr: PSR, | ||||
|  | ||||
|     /// Storage  for R8_fiq -> R12_fiq and their normal counterparts | ||||
|     /// e.g [r[0 + 8], fiq_r[0 + 8], r[1 + 8], fiq_r[1 + 8]...] | ||||
|     banked_fiq: [2 * 5]u32, | ||||
|  | ||||
|     /// Storage for r13_<mode>, r14_<mode> | ||||
|     /// e.g. [r13, r14, r13_svc, r14_svc] | ||||
|     banked_r: [2 * 6]u32, | ||||
|  | ||||
|     banked_spsr: [5]PSR, | ||||
|  | ||||
|     log_file: ?*const File, | ||||
|     log_buf: [0x100]u8, | ||||
|     binary_log: bool, | ||||
|  | ||||
|     pub fn init(alloc: Allocator, sched: *Scheduler, paths: FilePaths) !Self { | ||||
|         var cpu: Arm7tdmi = .{ | ||||
|             .r = [_]u32{0x00} ** 16, | ||||
|             .sched = sched, | ||||
|             .bus = try Bus.init(alloc, sched, paths), | ||||
|             .cpsr = .{ .raw = 0x0000_001F }, | ||||
|             .spsr = .{ .raw = 0x0000_0000 }, | ||||
|             .banked_fiq = [_]u32{0x00} ** 10, | ||||
|             .banked_r = [_]u32{0x00} ** 12, | ||||
|             .banked_spsr = [_]PSR{.{ .raw = 0x0000_0000 }} ** 5, | ||||
|             .log_file = null, | ||||
|             .log_buf = undefined, | ||||
|             .binary_log = false, | ||||
|         }; | ||||
|         cpu.bus.cpu = &cpu; | ||||
|         return cpu; | ||||
|     } | ||||
|  | ||||
|     pub fn deinit(self: Self) void { | ||||
|         self.bus.deinit(); | ||||
|     } | ||||
|  | ||||
|     pub fn useLogger(self: *Self, file: *const File, is_binary: bool) void { | ||||
|         self.log_file = file; | ||||
|         self.binary_log = is_binary; | ||||
|     } | ||||
|  | ||||
|     inline fn bankedIdx(mode: Mode, kind: BankedKind) usize { | ||||
|         const idx: usize = switch (mode) { | ||||
|             .User, .System => 0, | ||||
|             .Supervisor => 1, | ||||
|             .Abort => 2, | ||||
|             .Undefined => 3, | ||||
|             .Irq => 4, | ||||
|             .Fiq => 5, | ||||
|         }; | ||||
|  | ||||
|         return (idx * 2) + if (kind == .R14) @as(usize, 1) else 0; | ||||
|     } | ||||
|  | ||||
|     inline fn bankedSpsrIndex(mode: Mode) usize { | ||||
|         return switch (mode) { | ||||
|             .Supervisor => 0, | ||||
|             .Abort => 1, | ||||
|             .Undefined => 2, | ||||
|             .Irq => 3, | ||||
|             .Fiq => 4, | ||||
|             else => std.debug.panic("[CPU/Mode] {} does not have a SPSR Register", .{mode}), | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     inline fn bankedFiqIdx(i: usize, mode: Mode) usize { | ||||
|         return (i * 2) + if (mode == .Fiq) @as(usize, 1) else 0; | ||||
|     } | ||||
|  | ||||
|     pub inline fn hasSPSR(self: *const Self) bool { | ||||
|         const mode = getModeChecked(self, self.cpsr.mode.read()); | ||||
|         return switch (mode) { | ||||
|             .System, .User => false, | ||||
|             else => true, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub inline fn isPrivileged(self: *const Self) bool { | ||||
|         const mode = getModeChecked(self, self.cpsr.mode.read()); | ||||
|         return switch (mode) { | ||||
|             .User => false, | ||||
|             else => true, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn setCpsr(self: *Self, value: u32) void { | ||||
|         if (value & 0x1F != self.cpsr.raw & 0x1F) self.changeModeFromIdx(@truncate(u5, value & 0x1F)); | ||||
|         self.cpsr.raw = value; | ||||
|     } | ||||
|  | ||||
|     fn changeModeFromIdx(self: *Self, next: u5) void { | ||||
|         self.changeMode(getModeChecked(self, next)); | ||||
|     } | ||||
|  | ||||
|     pub fn setUserModeRegister(self: *Self, idx: usize, value: u32) void { | ||||
|         const current = getModeChecked(self, self.cpsr.mode.read()); | ||||
|  | ||||
|         switch (idx) { | ||||
|             8...12 => { | ||||
|                 if (current == .Fiq) { | ||||
|                     self.banked_fiq[bankedFiqIdx(idx - 8, .User)] = value; | ||||
|                 } else self.r[idx] = value; | ||||
|             }, | ||||
|             13, 14 => switch (current) { | ||||
|                 .User, .System => self.r[idx] = value, | ||||
|                 else => { | ||||
|                     const kind = std.meta.intToEnum(BankedKind, idx - 13) catch unreachable; | ||||
|                     self.banked_r[bankedIdx(.User, kind)] = value; | ||||
|                 }, | ||||
|             }, | ||||
|             else => self.r[idx] = value, // R0 -> R7  and R15 | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn getUserModeRegister(self: *Self, idx: usize) u32 { | ||||
|         const current = getModeChecked(self, self.cpsr.mode.read()); | ||||
|  | ||||
|         return switch (idx) { | ||||
|             8...12 => if (current == .Fiq) self.banked_fiq[bankedFiqIdx(idx - 8, .User)] else self.r[idx], | ||||
|             13, 14 => switch (current) { | ||||
|                 .User, .System => self.r[idx], | ||||
|                 else => blk: { | ||||
|                     const kind = std.meta.intToEnum(BankedKind, idx - 13) catch unreachable; | ||||
|                     break :blk self.banked_r[bankedIdx(.User, kind)]; | ||||
|                 }, | ||||
|             }, | ||||
|             else => self.r[idx], // R0 -> R7  and R15 | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn changeMode(self: *Self, next: Mode) void { | ||||
|         const now = getModeChecked(self, self.cpsr.mode.read()); | ||||
|  | ||||
|         // Bank R8 -> r12 | ||||
|         var i: usize = 0; | ||||
|         while (i < 5) : (i += 1) { | ||||
|             self.banked_fiq[bankedFiqIdx(i, now)] = self.r[8 + i]; | ||||
|         } | ||||
|  | ||||
|         // Bank r13, r14, SPSR | ||||
|         switch (now) { | ||||
|             .User, .System => { | ||||
|                 self.banked_r[bankedIdx(now, .R13)] = self.r[13]; | ||||
|                 self.banked_r[bankedIdx(now, .R14)] = self.r[14]; | ||||
|             }, | ||||
|             else => { | ||||
|                 self.banked_r[bankedIdx(now, .R13)] = self.r[13]; | ||||
|                 self.banked_r[bankedIdx(now, .R14)] = self.r[14]; | ||||
|                 self.banked_spsr[bankedSpsrIndex(now)] = self.spsr; | ||||
|             }, | ||||
|         } | ||||
|  | ||||
|         // Grab R8 -> R12 | ||||
|         i = 0; | ||||
|         while (i < 5) : (i += 1) { | ||||
|             self.r[8 + i] = self.banked_fiq[bankedFiqIdx(i, next)]; | ||||
|         } | ||||
|  | ||||
|         // Grab r13, r14, SPSR | ||||
|         switch (next) { | ||||
|             .User, .System => { | ||||
|                 self.r[13] = self.banked_r[bankedIdx(next, .R13)]; | ||||
|                 self.r[14] = self.banked_r[bankedIdx(next, .R14)]; | ||||
|             }, | ||||
|             else => { | ||||
|                 self.r[13] = self.banked_r[bankedIdx(next, .R13)]; | ||||
|                 self.r[14] = self.banked_r[bankedIdx(next, .R14)]; | ||||
|                 self.spsr = self.banked_spsr[bankedSpsrIndex(next)]; | ||||
|             }, | ||||
|         } | ||||
|  | ||||
|         self.cpsr.mode.write(@enumToInt(next)); | ||||
|     } | ||||
|  | ||||
|     pub fn fastBoot(self: *Self) void { | ||||
|         self.r = std.mem.zeroes([16]u32); | ||||
|  | ||||
|         self.r[0] = 0x08000000; | ||||
|         self.r[1] = 0x000000EA; | ||||
|         self.r[13] = 0x0300_7F00; | ||||
|         self.r[15] = 0x0800_0000; | ||||
|  | ||||
|         self.banked_r[bankedIdx(.Irq, .R13)] = 0x0300_7FA0; | ||||
|         self.banked_r[bankedIdx(.Supervisor, .R13)] = 0x0300_7FE0; | ||||
|  | ||||
|         self.cpsr.raw = 0x6000001F; | ||||
|     } | ||||
|  | ||||
|     pub fn step(self: *Self) void { | ||||
|         if (self.cpsr.t.read()) { | ||||
|             const opcode = self.thumbFetch(); | ||||
|             if (enable_logging) if (self.log_file) |file| self.debug_log(file, opcode); | ||||
|  | ||||
|             thumb_lut[thumbIdx(opcode)](self, &self.bus, opcode); | ||||
|         } else { | ||||
|             const opcode = self.fetch(); | ||||
|             if (enable_logging) if (self.log_file) |file| self.debug_log(file, opcode); | ||||
|  | ||||
|             if (checkCond(self.cpsr, @truncate(u4, opcode >> 28))) { | ||||
|                 arm_lut[armIdx(opcode)](self, &self.bus, opcode); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn handleInterrupt(self: *Self) void { | ||||
|         const should_handle = self.bus.io.ie.raw & self.bus.io.irq.raw; | ||||
|  | ||||
|         if (should_handle != 0) { | ||||
|             self.bus.io.haltcnt = .Execute; | ||||
|             // log.info("An Interrupt was Fired!", .{}); | ||||
|  | ||||
|             // Either IME is not true or I in CPSR is true | ||||
|             // Don't handle interrupts | ||||
|             if (!self.bus.io.ime or self.cpsr.i.read()) return; | ||||
|             // log.info("An interrupt was Handled!", .{}); | ||||
|  | ||||
|             // retAddr.gba says r15 on it's own is off by -04h in both ARM and THUMB mode | ||||
|             const r15 = self.r[15] + 4; | ||||
|             const cpsr = self.cpsr.raw; | ||||
|  | ||||
|             self.changeMode(.Irq); | ||||
|             self.cpsr.t.write(false); | ||||
|             self.cpsr.i.write(true); | ||||
|  | ||||
|             self.r[14] = r15; | ||||
|             self.spsr.raw = cpsr; | ||||
|             self.r[15] = 0x000_0018; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn thumbFetch(self: *Self) u16 { | ||||
|         defer self.r[15] += 2; | ||||
|         return self.bus.read(u16, self.r[15]); | ||||
|     } | ||||
|  | ||||
|     fn fetch(self: *Self) u32 { | ||||
|         defer self.r[15] += 4; | ||||
|         return self.bus.read(u32, self.r[15]); | ||||
|     } | ||||
|  | ||||
|     pub fn fakePC(self: *const Self) u32 { | ||||
|         return self.r[15] + 4; | ||||
|     } | ||||
|  | ||||
|     fn debug_log(self: *const Self, file: *const File, opcode: u32) void { | ||||
|         if (self.binary_log) { | ||||
|             self.skyLog(file) catch unreachable; | ||||
|         } else { | ||||
|             self.mgbaLog(file, opcode) catch unreachable; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn panic(self: *const Self, comptime format: []const u8, args: anytype) noreturn { | ||||
|         var i: usize = 0; | ||||
|         while (i < 16) : (i += 4) { | ||||
|             const i_1 = i + 1; | ||||
|             const i_2 = i + 2; | ||||
|             const i_3 = i + 3; | ||||
|             std.debug.print("R{}: 0x{X:0>8}\tR{}: 0x{X:0>8}\tR{}: 0x{X:0>8}\tR{}: 0x{X:0>8}\n", .{ i, self.r[i], i_1, self.r[i_1], i_2, self.r[i_2], i_3, self.r[i_3] }); | ||||
|         } | ||||
|         std.debug.print("cpsr: 0x{X:0>8} ", .{self.cpsr.raw}); | ||||
|         prettyPrintPsr(&self.cpsr); | ||||
|  | ||||
|         std.debug.print("spsr: 0x{X:0>8} ", .{self.spsr.raw}); | ||||
|         prettyPrintPsr(&self.spsr); | ||||
|  | ||||
|         if (self.cpsr.t.read()) { | ||||
|             const opcode = self.bus.read(u16, self.r[15] - 4); | ||||
|             const id = thumbIdx(opcode); | ||||
|             std.debug.print("opcode: ID: 0x{b:0>10} 0x{X:0>4}\n", .{ id, opcode }); | ||||
|         } else { | ||||
|             const opcode = self.bus.read(u32, self.r[15] - 4); | ||||
|             const id = armIdx(opcode); | ||||
|             std.debug.print("opcode: ID: 0x{X:0>3} 0x{X:0>8}\n", .{ id, opcode }); | ||||
|         } | ||||
|  | ||||
|         std.debug.print("tick: {}\n\n", .{self.sched.tick}); | ||||
|  | ||||
|         std.debug.panic(format, args); | ||||
|     } | ||||
|  | ||||
|     fn prettyPrintPsr(psr: *const PSR) void { | ||||
|         std.debug.print("[", .{}); | ||||
|  | ||||
|         if (psr.n.read()) std.debug.print("N", .{}) else std.debug.print("-", .{}); | ||||
|         if (psr.z.read()) std.debug.print("Z", .{}) else std.debug.print("-", .{}); | ||||
|         if (psr.c.read()) std.debug.print("C", .{}) else std.debug.print("-", .{}); | ||||
|         if (psr.v.read()) std.debug.print("V", .{}) else std.debug.print("-", .{}); | ||||
|         if (psr.i.read()) std.debug.print("I", .{}) else std.debug.print("-", .{}); | ||||
|         if (psr.f.read()) std.debug.print("F", .{}) else std.debug.print("-", .{}); | ||||
|         if (psr.t.read()) std.debug.print("T", .{}) else std.debug.print("-", .{}); | ||||
|         std.debug.print("|", .{}); | ||||
|         if (getMode(psr.mode.read())) |mode| std.debug.print("{s}", .{modeString(mode)}) else std.debug.print("---", .{}); | ||||
|  | ||||
|         std.debug.print("]\n", .{}); | ||||
|     } | ||||
|  | ||||
|     fn modeString(mode: Mode) []const u8 { | ||||
|         return switch (mode) { | ||||
|             .User => "usr", | ||||
|             .Fiq => "fiq", | ||||
|             .Irq => "irq", | ||||
|             .Supervisor => "svc", | ||||
|             .Abort => "abt", | ||||
|             .Undefined => "und", | ||||
|             .System => "sys", | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn skyLog(self: *const Self, file: *const File) !void { | ||||
|         var buf: [18 * @sizeOf(u32)]u8 = undefined; | ||||
|  | ||||
|         // Write Registers | ||||
|         var i: usize = 0; | ||||
|         while (i < 0x10) : (i += 1) { | ||||
|             skyWrite(&buf, i, self.r[i]); | ||||
|         } | ||||
|  | ||||
|         skyWrite(&buf, 0x10, self.cpsr.raw); | ||||
|         skyWrite(&buf, 0x11, if (self.hasSPSR()) self.spsr.raw else self.cpsr.raw); | ||||
|         _ = try file.writeAll(&buf); | ||||
|     } | ||||
|  | ||||
|     fn skyWrite(buf: []u8, i: usize, num: u32) void { | ||||
|         buf[(@sizeOf(u32) * i) + 3] = @truncate(u8, num >> 24 & 0xFF); | ||||
|         buf[(@sizeOf(u32) * i) + 2] = @truncate(u8, num >> 16 & 0xFF); | ||||
|         buf[(@sizeOf(u32) * i) + 1] = @truncate(u8, num >> 8 & 0xFF); | ||||
|         buf[(@sizeOf(u32) * i) + 0] = @truncate(u8, num >> 0 & 0xFF); | ||||
|     } | ||||
|  | ||||
|     fn mgbaLog(self: *const Self, file: *const File, opcode: u32) !void { | ||||
|         const thumb_fmt = "{X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} cpsr: {X:0>8} | {X:0>4}:\n"; | ||||
|         const arm_fmt = "{X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} cpsr: {X:0>8} | {X:0>8}:\n"; | ||||
|         var buf: [0x100]u8 = [_]u8{0x00} ** 0x100; // this is larger than it needs to be | ||||
|  | ||||
|         const r0 = self.r[0]; | ||||
|         const r1 = self.r[1]; | ||||
|         const r2 = self.r[2]; | ||||
|         const r3 = self.r[3]; | ||||
|         const r4 = self.r[4]; | ||||
|         const r5 = self.r[5]; | ||||
|         const r6 = self.r[6]; | ||||
|         const r7 = self.r[7]; | ||||
|         const r8 = self.r[8]; | ||||
|         const r9 = self.r[9]; | ||||
|         const r10 = self.r[10]; | ||||
|         const r11 = self.r[11]; | ||||
|         const r12 = self.r[12]; | ||||
|         const r13 = self.r[13]; | ||||
|         const r14 = self.r[14]; | ||||
|         const r15 = self.r[15]; | ||||
|  | ||||
|         const c_psr = self.cpsr.raw; | ||||
|  | ||||
|         var log_str: []u8 = undefined; | ||||
|         if (self.cpsr.t.read()) { | ||||
|             if (opcode >> 11 == 0x1E) { | ||||
|                 // Instruction 1 of a BL Opcode, print in ARM mode | ||||
|                 const tmp_opcode = self.bus.read(u32, self.r[15] - 2); | ||||
|                 const be_opcode = tmp_opcode << 16 | tmp_opcode >> 16; | ||||
|                 log_str = try std.fmt.bufPrint(&buf, arm_fmt, .{ r0, r1, r2, r3, r4, r5, r6, r7, r8, r9, r10, r11, r12, r13, r14, r15, c_psr, be_opcode }); | ||||
|             } else { | ||||
|                 log_str = try std.fmt.bufPrint(&buf, thumb_fmt, .{ r0, r1, r2, r3, r4, r5, r6, r7, r8, r9, r10, r11, r12, r13, r14, r15, c_psr, opcode }); | ||||
|             } | ||||
|         } else { | ||||
|             log_str = try std.fmt.bufPrint(&buf, arm_fmt, .{ r0, r1, r2, r3, r4, r5, r6, r7, r8, r9, r10, r11, r12, r13, r14, r15, c_psr, opcode }); | ||||
|         } | ||||
|  | ||||
|         _ = try file.writeAll(log_str); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| inline fn armIdx(opcode: u32) u12 { | ||||
|     return @truncate(u12, opcode >> 20 & 0xFF) << 4 | @truncate(u12, opcode >> 4 & 0xF); | ||||
| } | ||||
|  | ||||
| inline fn thumbIdx(opcode: u16) u10 { | ||||
|     return @truncate(u10, opcode >> 6); | ||||
| } | ||||
|  | ||||
| pub fn checkCond(cpsr: PSR, cond: u4) bool { | ||||
|     // TODO: Should I implement an enum? | ||||
|     return switch (cond) { | ||||
|         0x0 => cpsr.z.read(), // EQ - Equal | ||||
|         0x1 => !cpsr.z.read(), // NE - Not equal | ||||
|         0x2 => cpsr.c.read(), // CS - Unsigned higher or same | ||||
|         0x3 => !cpsr.c.read(), // CC - Unsigned lower | ||||
|         0x4 => cpsr.n.read(), // MI - Negative | ||||
|         0x5 => !cpsr.n.read(), // PL - Positive or zero | ||||
|         0x6 => cpsr.v.read(), // VS - Overflow | ||||
|         0x7 => !cpsr.v.read(), // VC - No overflow | ||||
|         0x8 => cpsr.c.read() and !cpsr.z.read(), // HI - unsigned higher | ||||
|         0x9 => !cpsr.c.read() or cpsr.z.read(), // LS - unsigned lower or same | ||||
|         0xA => cpsr.n.read() == cpsr.v.read(), // GE - Greater or equal | ||||
|         0xB => cpsr.n.read() != cpsr.v.read(), // LT - Less than | ||||
|         0xC => !cpsr.z.read() and (cpsr.n.read() == cpsr.v.read()), // GT - Greater than | ||||
|         0xD => cpsr.z.read() or (cpsr.n.read() != cpsr.v.read()), // LE - Less than or equal | ||||
|         0xE => true, // AL - Always | ||||
|         0xF => std.debug.panic("[CPU/Cond] 0xF is a reserved condition field", .{}), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| fn thumbPopulate() [0x400]ThumbInstrFn { | ||||
|     return comptime { | ||||
|         @setEvalBranchQuota(5025); // This is exact | ||||
|         var lut = [_]ThumbInstrFn{thumbUndefined} ** 0x400; | ||||
|  | ||||
|         var i: usize = 0; | ||||
|         while (i < lut.len) : (i += 1) { | ||||
|             lut[i] = switch (@as(u3, i >> 7 & 0x7)) { | ||||
|                 0b000 => if (i >> 5 & 0x3 == 0b11) blk: { | ||||
|                     const I = i >> 4 & 1 == 1; | ||||
|                     const is_sub = i >> 3 & 1 == 1; | ||||
|                     const rn = i & 0x7; | ||||
|                     break :blk format2(I, is_sub, rn); | ||||
|                 } else blk: { | ||||
|                     const op = i >> 5 & 0x3; | ||||
|                     const offset = i & 0x1F; | ||||
|                     break :blk format1(op, offset); | ||||
|                 }, | ||||
|                 0b001 => blk: { | ||||
|                     const op = i >> 5 & 0x3; | ||||
|                     const rd = i >> 2 & 0x7; | ||||
|                     break :blk format3(op, rd); | ||||
|                 }, | ||||
|                 0b010 => switch (@as(u2, i >> 5 & 0x3)) { | ||||
|                     0b00 => if (i >> 4 & 1 == 1) blk: { | ||||
|                         const op = i >> 2 & 0x3; | ||||
|                         const h1 = i >> 1 & 1; | ||||
|                         const h2 = i & 1; | ||||
|                         break :blk format5(op, h1, h2); | ||||
|                     } else blk: { | ||||
|                         const op = i & 0xF; | ||||
|                         break :blk format4(op); | ||||
|                     }, | ||||
|                     0b01 => blk: { | ||||
|                         const rd = i >> 2 & 0x7; | ||||
|                         break :blk format6(rd); | ||||
|                     }, | ||||
|                     else => blk: { | ||||
|                         const op = i >> 4 & 0x3; | ||||
|                         const T = i >> 3 & 1 == 1; | ||||
|                         break :blk format78(op, T); | ||||
|                     }, | ||||
|                 }, | ||||
|                 0b011 => blk: { | ||||
|                     const B = i >> 6 & 1 == 1; | ||||
|                     const L = i >> 5 & 1 == 1; | ||||
|                     const offset = i & 0x1F; | ||||
|                     break :blk format9(B, L, offset); | ||||
|                 }, | ||||
|                 else => switch (@as(u3, i >> 6 & 0x7)) { | ||||
|                     // MSB is guaranteed to be 1 | ||||
|                     0b000 => blk: { | ||||
|                         const L = i >> 5 & 1 == 1; | ||||
|                         const offset = i & 0x1F; | ||||
|                         break :blk format10(L, offset); | ||||
|                     }, | ||||
|                     0b001 => blk: { | ||||
|                         const L = i >> 5 & 1 == 1; | ||||
|                         const rd = i >> 2 & 0x7; | ||||
|                         break :blk format11(L, rd); | ||||
|                     }, | ||||
|                     0b010 => blk: { | ||||
|                         const isSP = i >> 5 & 1 == 1; | ||||
|                         const rd = i >> 2 & 0x7; | ||||
|                         break :blk format12(isSP, rd); | ||||
|                     }, | ||||
|                     0b011 => if (i >> 4 & 1 == 1) blk: { | ||||
|                         const L = i >> 5 & 1 == 1; | ||||
|                         const R = i >> 2 & 1 == 1; | ||||
|                         break :blk format14(L, R); | ||||
|                     } else blk: { | ||||
|                         const S = i >> 1 & 1 == 1; | ||||
|                         break :blk format13(S); | ||||
|                     }, | ||||
|                     0b100 => blk: { | ||||
|                         const L = i >> 5 & 1 == 1; | ||||
|                         const rb = i >> 2 & 0x7; | ||||
|  | ||||
|                         break :blk format15(L, rb); | ||||
|                     }, | ||||
|                     0b101 => if (i >> 2 & 0xF == 0b1111) blk: { | ||||
|                         break :blk thumbSoftwareInterrupt(); | ||||
|                     } else blk: { | ||||
|                         const cond = i >> 2 & 0xF; | ||||
|                         break :blk format16(cond); | ||||
|                     }, | ||||
|                     0b110 => format18(), | ||||
|                     0b111 => blk: { | ||||
|                         const is_low = i >> 5 & 1 == 1; | ||||
|                         break :blk format19(is_low); | ||||
|                     }, | ||||
|                 }, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         return lut; | ||||
|     }; | ||||
| } | ||||
|  | ||||
| fn armPopulate() [0x1000]ArmInstrFn { | ||||
|     return comptime { | ||||
|         @setEvalBranchQuota(0xE000); | ||||
|         var lut = [_]ArmInstrFn{armUndefined} ** 0x1000; | ||||
|  | ||||
|         var i: usize = 0; | ||||
|         while (i < lut.len) : (i += 1) { | ||||
|             lut[i] = switch (@as(u2, i >> 10)) { | ||||
|                 0b00 => if (i == 0x121) blk: { | ||||
|                     break :blk branchAndExchange; | ||||
|                 } else if (i & 0xFCF == 0x009) blk: { | ||||
|                     const A = i >> 5 & 1 == 1; | ||||
|                     const S = i >> 4 & 1 == 1; | ||||
|                     break :blk multiply(A, S); | ||||
|                 } else if (i & 0xFBF == 0x109) blk: { | ||||
|                     const B = i >> 6 & 1 == 1; | ||||
|                     break :blk singleDataSwap(B); | ||||
|                 } else if (i & 0xF8F == 0x089) blk: { | ||||
|                     const U = i >> 6 & 1 == 1; | ||||
|                     const A = i >> 5 & 1 == 1; | ||||
|                     const S = i >> 4 & 1 == 1; | ||||
|                     break :blk multiplyLong(U, A, S); | ||||
|                 } else if (i & 0xE49 == 0x009 or i & 0xE49 == 0x049) blk: { | ||||
|                     const P = i >> 8 & 1 == 1; | ||||
|                     const U = i >> 7 & 1 == 1; | ||||
|                     const I = i >> 6 & 1 == 1; | ||||
|                     const W = i >> 5 & 1 == 1; | ||||
|                     const L = i >> 4 & 1 == 1; | ||||
|                     break :blk halfAndSignedDataTransfer(P, U, I, W, L); | ||||
|                 } else if (i & 0xD90 == 0x100) blk: { | ||||
|                     const I = i >> 9 & 1 == 1; | ||||
|                     const R = i >> 6 & 1 == 1; | ||||
|                     const kind = i >> 4 & 0x3; | ||||
|                     break :blk psrTransfer(I, R, kind); | ||||
|                 } else blk: { | ||||
|                     const I = i >> 9 & 1 == 1; | ||||
|                     const S = i >> 4 & 1 == 1; | ||||
|                     const instrKind = i >> 5 & 0xF; | ||||
|                     break :blk dataProcessing(I, S, instrKind); | ||||
|                 }, | ||||
|                 0b01 => if (i >> 9 & 1 == 1 and i & 1 == 1) armUndefined else blk: { | ||||
|                     const I = i >> 9 & 1 == 1; | ||||
|                     const P = i >> 8 & 1 == 1; | ||||
|                     const U = i >> 7 & 1 == 1; | ||||
|                     const B = i >> 6 & 1 == 1; | ||||
|                     const W = i >> 5 & 1 == 1; | ||||
|                     const L = i >> 4 & 1 == 1; | ||||
|                     break :blk singleDataTransfer(I, P, U, B, W, L); | ||||
|                 }, | ||||
|                 else => switch (@as(u2, i >> 9 & 0x3)) { | ||||
|                     // MSB is guaranteed to be 1 | ||||
|                     0b00 => blk: { | ||||
|                         const P = i >> 8 & 1 == 1; | ||||
|                         const U = i >> 7 & 1 == 1; | ||||
|                         const S = i >> 6 & 1 == 1; | ||||
|                         const W = i >> 5 & 1 == 1; | ||||
|                         const L = i >> 4 & 1 == 1; | ||||
|                         break :blk blockDataTransfer(P, U, S, W, L); | ||||
|                     }, | ||||
|                     0b01 => blk: { | ||||
|                         const L = i >> 8 & 1 == 1; | ||||
|                         break :blk branch(L); | ||||
|                     }, | ||||
|                     0b10 => armUndefined, // COP Data Transfer | ||||
|                     0b11 => if (i >> 8 & 1 == 1) armSoftwareInterrupt() else armUndefined, // COP Data Operation + Register Transfer | ||||
|                 }, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         return lut; | ||||
|     }; | ||||
| } | ||||
|  | ||||
| pub const PSR = extern union { | ||||
|     mode: Bitfield(u32, 0, 5), | ||||
|     t: Bit(u32, 5), | ||||
|     f: Bit(u32, 6), | ||||
|     i: Bit(u32, 7), | ||||
|     v: Bit(u32, 28), | ||||
|     c: Bit(u32, 29), | ||||
|     z: Bit(u32, 30), | ||||
|     n: Bit(u32, 31), | ||||
|     raw: u32, | ||||
| }; | ||||
|  | ||||
| const Mode = enum(u5) { | ||||
|     User = 0b10000, | ||||
|     Fiq = 0b10001, | ||||
|     Irq = 0b10010, | ||||
|     Supervisor = 0b10011, | ||||
|     Abort = 0b10111, | ||||
|     Undefined = 0b11011, | ||||
|     System = 0b11111, | ||||
| }; | ||||
|  | ||||
| const BankedKind = enum(u1) { | ||||
|     R13 = 0, | ||||
|     R14, | ||||
| }; | ||||
|  | ||||
| fn getMode(bits: u5) ?Mode { | ||||
|     return std.meta.intToEnum(Mode, bits) catch null; | ||||
| } | ||||
|  | ||||
| fn getModeChecked(cpu: *const Arm7tdmi, bits: u5) Mode { | ||||
|     return getMode(bits) orelse cpu.panic("[CPU/CPSR] 0b{b:0>5} is an invalid CPU mode", .{bits}); | ||||
| } | ||||
|  | ||||
| fn armUndefined(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { | ||||
|     const id = armIdx(opcode); | ||||
|     cpu.panic("[CPU/Decode] ID: 0x{X:0>3} 0x{X:0>8} is an illegal opcode", .{ id, opcode }); | ||||
| } | ||||
|  | ||||
| fn thumbUndefined(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|     const id = thumbIdx(opcode); | ||||
|     cpu.panic("[CPU/Decode] ID: 0b{b:0>10} 0x{X:0>2} is an illegal opcode", .{ id, opcode }); | ||||
| } | ||||
| @@ -1,6 +1,6 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ArmInstrFn; | ||||
| 
 | ||||
| pub fn blockDataTransfer(comptime P: bool, comptime U: bool, comptime S: bool, comptime W: bool, comptime L: bool) InstrFn { | ||||
|     return struct { | ||||
| @@ -55,9 +55,8 @@ pub fn blockDataTransfer(comptime P: bool, comptime U: bool, comptime S: bool, c | ||||
| 
 | ||||
|                 if (L) { | ||||
|                     cpu.r[15] = bus.read(u32, und_addr); | ||||
|                     cpu.pipe.reload(cpu); | ||||
|                 } else { | ||||
|                     bus.write(u32, und_addr, cpu.r[15] + 4); | ||||
|                     bus.write(u32, und_addr, cpu.r[15] + 8); | ||||
|                 } | ||||
| 
 | ||||
|                 cpu.r[rn] = if (U) cpu.r[rn] + 0x40 else cpu.r[rn] - 0x40; | ||||
| @@ -87,23 +86,17 @@ pub fn blockDataTransfer(comptime P: bool, comptime U: bool, comptime S: bool, c | ||||
|                     cpu.setUserModeRegister(i, bus.read(u32, address)); | ||||
|                 } else { | ||||
|                     const value = bus.read(u32, address); | ||||
| 
 | ||||
|                     cpu.r[i] = value; | ||||
|                     if (i == 0xF) { | ||||
|                         cpu.r[i] &= ~@as(u32, 3); // Align r15 | ||||
|                         cpu.pipe.reload(cpu); | ||||
| 
 | ||||
|                         if (S) cpu.setCpsr(cpu.spsr.raw); | ||||
|                     } | ||||
|                     cpu.r[i] = if (i == 0xF) value & 0xFFFF_FFFC else value; | ||||
|                     if (S and i == 0xF) cpu.setCpsr(cpu.spsr.raw); | ||||
|                 } | ||||
|             } else { | ||||
|                 if (S) { | ||||
|                     // Always Transfer User mode Registers | ||||
|                     // This happens regardless if r15 is in the list | ||||
|                     const value = cpu.getUserModeRegister(i); | ||||
|                     bus.write(u32, address, value + if (i == 0xF) 4 else @as(u32, 0)); // PC is already 8 ahead to make 12 | ||||
|                     bus.write(u32, address, value + if (i == 0xF) 8 else @as(u32, 0)); // PC is already 4 ahead to make 12 | ||||
|                 } else { | ||||
|                     bus.write(u32, address, cpu.r[i] + if (i == 0xF) 4 else @as(u32, 0)); | ||||
|                     bus.write(u32, address, cpu.r[i] + if (i == 0xF) 8 else @as(u32, 0)); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
							
								
								
									
										22
									
								
								src/cpu/arm/branch.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/cpu/arm/branch.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").ArmInstrFn; | ||||
|  | ||||
| const sext = @import("../../util.zig").sext; | ||||
|  | ||||
| pub fn branch(comptime L: bool) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { | ||||
|             if (L) cpu.r[14] = cpu.r[15]; | ||||
|             cpu.r[15] = cpu.fakePC() +% sext(24, opcode << 2); | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn branchAndExchange(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { | ||||
|     const rn = opcode & 0xF; | ||||
|     cpu.cpsr.t.write(cpu.r[rn] & 1 == 1); | ||||
|     cpu.r[15] = cpu.r[rn] & 0xFFFF_FFFE; | ||||
| } | ||||
							
								
								
									
										284
									
								
								src/cpu/arm/data_processing.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										284
									
								
								src/cpu/arm/data_processing.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,284 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").ArmInstrFn; | ||||
|  | ||||
| const rotateRight = @import("../barrel_shifter.zig").rotateRight; | ||||
| const execute = @import("../barrel_shifter.zig").execute; | ||||
|  | ||||
| pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime instrKind: u4) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { | ||||
|             const rd = @truncate(u4, opcode >> 12 & 0xF); | ||||
|             const rn = opcode >> 16 & 0xF; | ||||
|             const old_carry = @boolToInt(cpu.cpsr.c.read()); | ||||
|  | ||||
|             // If certain conditions are met, PC is 12 ahead instead of 8 | ||||
|             if (!I and opcode >> 4 & 1 == 1) cpu.r[15] += 4; | ||||
|  | ||||
|             const op1 = if (rn == 0xF) cpu.fakePC() else cpu.r[rn]; | ||||
|  | ||||
|             var op2: u32 = undefined; | ||||
|             if (I) { | ||||
|                 const amount = @truncate(u8, (opcode >> 8 & 0xF) << 1); | ||||
|                 op2 = rotateRight(S, &cpu.cpsr, opcode & 0xFF, amount); | ||||
|             } else { | ||||
|                 op2 = execute(S, cpu, opcode); | ||||
|             } | ||||
|  | ||||
|             // Undo special condition from above | ||||
|             if (!I and opcode >> 4 & 1 == 1) cpu.r[15] -= 4; | ||||
|  | ||||
|             switch (instrKind) { | ||||
|                 0x0 => { | ||||
|                     // AND | ||||
|                     const result = op1 & op2; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setArmLogicOpFlags(S, cpu, rd, result); | ||||
|                 }, | ||||
|                 0x1 => { | ||||
|                     // EOR | ||||
|                     const result = op1 ^ op2; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setArmLogicOpFlags(S, cpu, rd, result); | ||||
|                 }, | ||||
|                 0x2 => { | ||||
|                     // SUB | ||||
|                     cpu.r[rd] = armSub(S, cpu, rd, op1, op2); | ||||
|                 }, | ||||
|                 0x3 => { | ||||
|                     // RSB | ||||
|                     cpu.r[rd] = armSub(S, cpu, rd, op2, op1); | ||||
|                 }, | ||||
|                 0x4 => { | ||||
|                     // ADD | ||||
|                     cpu.r[rd] = armAdd(S, cpu, rd, op1, op2); | ||||
|                 }, | ||||
|                 0x5 => { | ||||
|                     // ADC | ||||
|                     cpu.r[rd] = armAdc(S, cpu, rd, op1, op2, old_carry); | ||||
|                 }, | ||||
|                 0x6 => { | ||||
|                     // SBC | ||||
|                     cpu.r[rd] = armSbc(S, cpu, rd, op1, op2, old_carry); | ||||
|                 }, | ||||
|                 0x7 => { | ||||
|                     // RSC | ||||
|                     cpu.r[rd] = armSbc(S, cpu, rd, op2, op1, old_carry); | ||||
|                 }, | ||||
|                 0x8 => { | ||||
|                     // TST | ||||
|                     if (rd == 0xF) { | ||||
|                         undefinedTestBehaviour(cpu); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
|                     const result = op1 & op2; | ||||
|                     setTestOpFlags(S, cpu, opcode, result); | ||||
|                 }, | ||||
|                 0x9 => { | ||||
|                     // TEQ | ||||
|                     if (rd == 0xF) { | ||||
|                         undefinedTestBehaviour(cpu); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
|                     const result = op1 ^ op2; | ||||
|                     setTestOpFlags(S, cpu, opcode, result); | ||||
|                 }, | ||||
|                 0xA => { | ||||
|                     // CMP | ||||
|                     if (rd == 0xF) { | ||||
|                         undefinedTestBehaviour(cpu); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
|                     cmp(cpu, op1, op2); | ||||
|                 }, | ||||
|                 0xB => { | ||||
|                     // CMN | ||||
|                     if (rd == 0xF) { | ||||
|                         undefinedTestBehaviour(cpu); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
|                     cmn(cpu, op1, op2); | ||||
|                 }, | ||||
|                 0xC => { | ||||
|                     // ORR | ||||
|                     const result = op1 | op2; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setArmLogicOpFlags(S, cpu, rd, result); | ||||
|                 }, | ||||
|                 0xD => { | ||||
|                     // MOV | ||||
|                     cpu.r[rd] = op2; | ||||
|                     setArmLogicOpFlags(S, cpu, rd, op2); | ||||
|                 }, | ||||
|                 0xE => { | ||||
|                     // BIC | ||||
|                     const result = op1 & ~op2; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setArmLogicOpFlags(S, cpu, rd, result); | ||||
|                 }, | ||||
|                 0xF => { | ||||
|                     // MVN | ||||
|                     const result = ~op2; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setArmLogicOpFlags(S, cpu, rd, result); | ||||
|                 }, | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| fn armSbc(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32, old_carry: u1) u32 { | ||||
|     var result: u32 = undefined; | ||||
|     if (S and rd == 0xF) { | ||||
|         result = sbc(false, cpu, left, right, old_carry); | ||||
|         cpu.setCpsr(cpu.spsr.raw); | ||||
|     } else { | ||||
|         result = sbc(S, cpu, left, right, old_carry); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| pub fn sbc(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32, old_carry: u1) u32 { | ||||
|     // TODO: Make your own version (thanks peach.bot) | ||||
|     const subtrahend = @as(u64, right) -% old_carry +% 1; | ||||
|     const result = @truncate(u32, left -% subtrahend); | ||||
|  | ||||
|     if (S) { | ||||
|         cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|         cpu.cpsr.z.write(result == 0); | ||||
|         cpu.cpsr.c.write(subtrahend <= left); | ||||
|         cpu.cpsr.v.write(((left ^ result) & (~right ^ result)) >> 31 & 1 == 1); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| fn armSub(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32) u32 { | ||||
|     var result: u32 = undefined; | ||||
|     if (S and rd == 0xF) { | ||||
|         result = sub(false, cpu, left, right); | ||||
|         cpu.setCpsr(cpu.spsr.raw); | ||||
|     } else { | ||||
|         result = sub(S, cpu, left, right); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| pub fn sub(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32) u32 { | ||||
|     const result = left -% right; | ||||
|  | ||||
|     if (S) { | ||||
|         cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|         cpu.cpsr.z.write(result == 0); | ||||
|         cpu.cpsr.c.write(right <= left); | ||||
|         cpu.cpsr.v.write(((left ^ result) & (~right ^ result)) >> 31 & 1 == 1); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| fn armAdd(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32) u32 { | ||||
|     var result: u32 = undefined; | ||||
|     if (S and rd == 0xF) { | ||||
|         result = add(false, cpu, left, right); | ||||
|         cpu.setCpsr(cpu.spsr.raw); | ||||
|     } else { | ||||
|         result = add(S, cpu, left, right); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| pub fn add(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32) u32 { | ||||
|     var result: u32 = undefined; | ||||
|     const didOverflow = @addWithOverflow(u32, left, right, &result); | ||||
|  | ||||
|     if (S) { | ||||
|         cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|         cpu.cpsr.z.write(result == 0); | ||||
|         cpu.cpsr.c.write(didOverflow); | ||||
|         cpu.cpsr.v.write(((left ^ result) & (right ^ result)) >> 31 & 1 == 1); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| fn armAdc(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32, old_carry: u1) u32 { | ||||
|     var result: u32 = undefined; | ||||
|     if (S and rd == 0xF) { | ||||
|         result = adc(false, cpu, left, right, old_carry); | ||||
|         cpu.setCpsr(cpu.spsr.raw); | ||||
|     } else { | ||||
|         result = adc(S, cpu, left, right, old_carry); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| pub fn adc(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32, old_carry: u1) u32 { | ||||
|     var result: u32 = undefined; | ||||
|     const did = @addWithOverflow(u32, left, right, &result); | ||||
|     const overflow = @addWithOverflow(u32, result, old_carry, &result); | ||||
|  | ||||
|     if (S) { | ||||
|         cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|         cpu.cpsr.z.write(result == 0); | ||||
|         cpu.cpsr.c.write(did or overflow); | ||||
|         cpu.cpsr.v.write(((left ^ result) & (right ^ result)) >> 31 & 1 == 1); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| pub fn cmp(cpu: *Arm7tdmi, left: u32, right: u32) void { | ||||
|     const result = left -% right; | ||||
|  | ||||
|     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|     cpu.cpsr.z.write(result == 0); | ||||
|     cpu.cpsr.c.write(right <= left); | ||||
|     cpu.cpsr.v.write(((left ^ result) & (~right ^ result)) >> 31 & 1 == 1); | ||||
| } | ||||
|  | ||||
| pub fn cmn(cpu: *Arm7tdmi, left: u32, right: u32) void { | ||||
|     var result: u32 = undefined; | ||||
|     const didOverflow = @addWithOverflow(u32, left, right, &result); | ||||
|  | ||||
|     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|     cpu.cpsr.z.write(result == 0); | ||||
|     cpu.cpsr.c.write(didOverflow); | ||||
|     cpu.cpsr.v.write(((left ^ result) & (right ^ result)) >> 31 & 1 == 1); | ||||
| } | ||||
|  | ||||
| fn setArmLogicOpFlags(comptime S: bool, cpu: *Arm7tdmi, rd: u4, result: u32) void { | ||||
|     if (S and rd == 0xF) { | ||||
|         cpu.setCpsr(cpu.spsr.raw); | ||||
|     } else { | ||||
|         setLogicOpFlags(S, cpu, result); | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn setLogicOpFlags(comptime S: bool, cpu: *Arm7tdmi, result: u32) void { | ||||
|     if (S) { | ||||
|         cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|         cpu.cpsr.z.write(result == 0); | ||||
|         // C set by Barrel Shifter, V is unaffected | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn setTestOpFlags(comptime S: bool, cpu: *Arm7tdmi, opcode: u32, result: u32) void { | ||||
|     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|     cpu.cpsr.z.write(result == 0); | ||||
|     // Barrel Shifter should always calc CPSR C in TST | ||||
|     if (!S) _ = execute(true, cpu, opcode); | ||||
| } | ||||
|  | ||||
| fn undefinedTestBehaviour(cpu: *Arm7tdmi) void { | ||||
|     @setCold(true); | ||||
|     cpu.setCpsr(cpu.spsr.raw); | ||||
| } | ||||
| @@ -1,9 +1,11 @@ | ||||
| const std = @import("std"); | ||||
| 
 | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ArmInstrFn; | ||||
| 
 | ||||
| const sext = @import("../../../util.zig").sext; | ||||
| const rotr = @import("../../../util.zig").rotr; | ||||
| const sext = @import("../../util.zig").sext; | ||||
| const rotr = @import("../../util.zig").rotr; | ||||
| 
 | ||||
| pub fn halfAndSignedDataTransfer(comptime P: bool, comptime U: bool, comptime I: bool, comptime W: bool, comptime L: bool) InstrFn { | ||||
|     return struct { | ||||
| @@ -13,8 +15,20 @@ pub fn halfAndSignedDataTransfer(comptime P: bool, comptime U: bool, comptime I: | ||||
|             const rm = opcode & 0xF; | ||||
|             const imm_offset_high = opcode >> 8 & 0xF; | ||||
| 
 | ||||
|             const base = cpu.r[rn] + if (!L and rn == 0xF) 4 else @as(u32, 0); | ||||
|             const offset = if (I) imm_offset_high << 4 | rm else cpu.r[rm]; | ||||
|             var base: u32 = undefined; | ||||
|             if (rn == 0xF) { | ||||
|                 base = cpu.fakePC(); | ||||
|                 if (!L) base += 4; | ||||
|             } else { | ||||
|                 base = cpu.r[rn]; | ||||
|             } | ||||
| 
 | ||||
|             var offset: u32 = undefined; | ||||
|             if (I) { | ||||
|                 offset = imm_offset_high << 4 | rm; | ||||
|             } else { | ||||
|                 offset = cpu.r[rm]; | ||||
|             } | ||||
| 
 | ||||
|             const modified_base = if (U) base +% offset else base -% offset; | ||||
|             var address = if (P) modified_base else base; | ||||
| @@ -29,12 +43,15 @@ pub fn halfAndSignedDataTransfer(comptime P: bool, comptime U: bool, comptime I: | ||||
|                     }, | ||||
|                     0b10 => { | ||||
|                         // LDRSB | ||||
|                         result = sext(u32, u8, bus.read(u8, address)); | ||||
|                         result = sext(8, bus.read(u8, address)); | ||||
|                     }, | ||||
|                     0b11 => { | ||||
|                         // LDRSH | ||||
|                         const value = bus.read(u16, address); | ||||
|                         result = if (address & 1 == 1) sext(u32, u8, @truncate(u8, value >> 8)) else sext(u32, u16, value); | ||||
|                         result = if (address & 1 == 1) blk: { | ||||
|                             break :blk sext(8, bus.read(u8, address)); | ||||
|                         } else blk: { | ||||
|                             break :blk sext(16, bus.read(u16, address)); | ||||
|                         }; | ||||
|                     }, | ||||
|                     0b00 => unreachable, // SWP | ||||
|                 } | ||||
| @@ -1,6 +1,6 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ArmInstrFn; | ||||
| 
 | ||||
| pub fn multiply(comptime A: bool, comptime S: bool) InstrFn { | ||||
|     return struct { | ||||
| @@ -2,12 +2,12 @@ const std = @import("std"); | ||||
| 
 | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ArmInstrFn; | ||||
| const PSR = @import("../../cpu.zig").PSR; | ||||
| 
 | ||||
| const log = std.log.scoped(.PsrTransfer); | ||||
| 
 | ||||
| const rotr = @import("../../../util.zig").rotr; | ||||
| const rotr = @import("../../util.zig").rotr; | ||||
| 
 | ||||
| pub fn psrTransfer(comptime I: bool, comptime R: bool, comptime kind: u2) InstrFn { | ||||
|     return struct { | ||||
| @@ -17,22 +17,19 @@ pub fn psrTransfer(comptime I: bool, comptime R: bool, comptime kind: u2) InstrF | ||||
|                     // MRS | ||||
|                     const rd = opcode >> 12 & 0xF; | ||||
| 
 | ||||
|                     if (R and !cpu.hasSPSR()) log.err("Tried to read SPSR from User/System Mode", .{}); | ||||
|                     if (R and !cpu.hasSPSR()) log.warn("Tried to read SPSR from User/System Mode", .{}); | ||||
|                     cpu.r[rd] = if (R) cpu.spsr.raw else cpu.cpsr.raw; | ||||
|                 }, | ||||
|                 0b10 => { | ||||
|                     // MSR | ||||
|                     const field_mask = @truncate(u4, opcode >> 16 & 0xF); | ||||
|                     const rm_idx = opcode & 0xF; | ||||
|                     const right = if (I) rotr(u32, opcode & 0xFF, (opcode >> 8 & 0xF) * 2) else cpu.r[rm_idx]; | ||||
|                     const right = if (I) rotr(u32, opcode & 0xFF, (opcode >> 8 & 0xF) << 1) else cpu.r[rm_idx]; | ||||
| 
 | ||||
|                     if (R and !cpu.hasSPSR()) log.err("Tried to write to SPSR in User/System Mode", .{}); | ||||
|                     if (R and !cpu.hasSPSR()) log.warn("Tried to write to SPSR in User/System Mode", .{}); | ||||
| 
 | ||||
|                     if (R) { | ||||
|                         // arm.gba seems to expect the SPSR to do somethign in SYS mode, | ||||
|                         // so we just assume that despite writing to the SPSR in USR or SYS mode | ||||
|                         // being UNPREDICTABLE, it just magically has a working SPSR somehow | ||||
|                         cpu.spsr.raw = fieldMask(&cpu.spsr, field_mask, right); | ||||
|                         if (cpu.isPrivileged()) cpu.spsr.raw = fieldMask(&cpu.spsr, field_mask, right); | ||||
|                     } else { | ||||
|                         if (cpu.isPrivileged()) cpu.setCpsr(fieldMask(&cpu.cpsr, field_mask, right)); | ||||
|                     } | ||||
| @@ -44,8 +41,6 @@ pub fn psrTransfer(comptime I: bool, comptime R: bool, comptime kind: u2) InstrF | ||||
| } | ||||
| 
 | ||||
| fn fieldMask(psr: *const PSR, field_mask: u4, right: u32) u32 { | ||||
|     // This bitwise ORs bits 3 and 0 of the field mask into a u2 | ||||
|     // We do this because we only care about bits 7:0 and 31:28 of the CPSR | ||||
|     const bits = @truncate(u2, (field_mask >> 2 & 0x2) | (field_mask & 1)); | ||||
| 
 | ||||
|     const mask: u32 = switch (bits) { | ||||
| @@ -1,8 +1,10 @@ | ||||
| const std = @import("std"); | ||||
| 
 | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ArmInstrFn; | ||||
| 
 | ||||
| const rotr = @import("../../../util.zig").rotr; | ||||
| const rotr = @import("../../util.zig").rotr; | ||||
| 
 | ||||
| pub fn singleDataSwap(comptime B: bool) InstrFn { | ||||
|     return struct { | ||||
| @@ -1,9 +1,12 @@ | ||||
| const std = @import("std"); | ||||
| const util = @import("../../util.zig"); | ||||
| 
 | ||||
| const shifter = @import("../barrel_shifter.zig"); | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ArmInstrFn; | ||||
| 
 | ||||
| const rotr = @import("../../../util.zig").rotr; | ||||
| const rotr = @import("../../util.zig").rotr; | ||||
| 
 | ||||
| pub fn singleDataTransfer(comptime I: bool, comptime P: bool, comptime U: bool, comptime B: bool, comptime W: bool, comptime L: bool) InstrFn { | ||||
|     return struct { | ||||
| @@ -11,10 +14,15 @@ pub fn singleDataTransfer(comptime I: bool, comptime P: bool, comptime U: bool, | ||||
|             const rn = opcode >> 16 & 0xF; | ||||
|             const rd = opcode >> 12 & 0xF; | ||||
| 
 | ||||
|             // rn is r15 and L is not set, the PC is 12 ahead | ||||
|             const base = cpu.r[rn] + if (!L and rn == 0xF) 4 else @as(u32, 0); | ||||
|             var base: u32 = undefined; | ||||
|             if (rn == 0xF) { | ||||
|                 base = cpu.fakePC(); | ||||
|                 if (!L) base += 4; // Offset of 12 | ||||
|             } else { | ||||
|                 base = cpu.r[rn]; | ||||
|             } | ||||
| 
 | ||||
|             const offset = if (I) shifter.immediate(false, cpu, opcode) else opcode & 0xFFF; | ||||
|             const offset = if (I) shifter.immShift(false, cpu, opcode) else opcode & 0xFFF; | ||||
| 
 | ||||
|             const modified_base = if (U) base +% offset else base -% offset; | ||||
|             var address = if (P) modified_base else base; | ||||
| @@ -32,26 +40,18 @@ pub fn singleDataTransfer(comptime I: bool, comptime P: bool, comptime U: bool, | ||||
|             } else { | ||||
|                 if (B) { | ||||
|                     // STRB | ||||
|                     const value = cpu.r[rd] + if (rd == 0xF) 4 else @as(u32, 0); // PC is 12 ahead | ||||
|                     const value = if (rd == 0xF) cpu.r[rd] + 8 else cpu.r[rd]; | ||||
|                     bus.write(u8, address, @truncate(u8, value)); | ||||
|                 } else { | ||||
|                     // STR | ||||
|                     const value = cpu.r[rd] + if (rd == 0xF) 4 else @as(u32, 0); | ||||
|                     const value = if (rd == 0xF) cpu.r[rd] + 8 else cpu.r[rd]; | ||||
|                     bus.write(u32, address, value); | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             address = modified_base; | ||||
|             if (W and P or !P) { | ||||
|                 cpu.r[rn] = address; | ||||
|                 if (rn == 0xF) cpu.pipe.reload(cpu); | ||||
|             } | ||||
| 
 | ||||
|             if (L) { | ||||
|                 // This emulates the LDR rd == rn behaviour | ||||
|                 cpu.r[rd] = result; | ||||
|                 if (rd == 0xF) cpu.pipe.reload(cpu); | ||||
|             } | ||||
|             if (W and P or !P) cpu.r[rn] = address; | ||||
|             if (L) cpu.r[rd] = result; // This emulates the LDR rd == rn behaviour | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
| @@ -1,12 +1,12 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ArmInstrFn; | ||||
| 
 | ||||
| pub fn armSoftwareInterrupt() InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, _: u32) void { | ||||
|             // Copy Values from Current Mode | ||||
|             const ret_addr = cpu.r[15] - 4; | ||||
|             const r15 = cpu.r[15]; | ||||
|             const cpsr = cpu.cpsr.raw; | ||||
| 
 | ||||
|             // Switch Mode | ||||
| @@ -14,10 +14,9 @@ pub fn armSoftwareInterrupt() InstrFn { | ||||
|             cpu.cpsr.t.write(false); // Force ARM Mode | ||||
|             cpu.cpsr.i.write(true); // Disable normal interrupts | ||||
| 
 | ||||
|             cpu.r[14] = ret_addr; // Resume Execution | ||||
|             cpu.r[14] = r15; // Resume Execution | ||||
|             cpu.spsr.raw = cpsr; // Previous mode CPSR | ||||
|             cpu.r[15] = 0x0000_0008; | ||||
|             cpu.pipe.reload(cpu); | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
| @@ -1,35 +1,41 @@ | ||||
| const std = @import("std"); | ||||
| 
 | ||||
| const Arm7tdmi = @import("../cpu.zig").Arm7tdmi; | ||||
| const CPSR = @import("../cpu.zig").PSR; | ||||
| 
 | ||||
| const rotr = @import("../../util.zig").rotr; | ||||
| const rotr = @import("../util.zig").rotr; | ||||
| 
 | ||||
| pub fn exec(comptime S: bool, cpu: *Arm7tdmi, opcode: u32) u32 { | ||||
| pub fn execute(comptime S: bool, cpu: *Arm7tdmi, opcode: u32) u32 { | ||||
|     var result: u32 = undefined; | ||||
|     if (opcode >> 4 & 1 == 1) { | ||||
|         result = register(S, cpu, opcode); | ||||
|         result = registerShift(S, cpu, opcode); | ||||
|     } else { | ||||
|         result = immediate(S, cpu, opcode); | ||||
|         result = immShift(S, cpu, opcode); | ||||
|     } | ||||
| 
 | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| fn register(comptime S: bool, cpu: *Arm7tdmi, opcode: u32) u32 { | ||||
| fn registerShift(comptime S: bool, cpu: *Arm7tdmi, opcode: u32) u32 { | ||||
|     const rs_idx = opcode >> 8 & 0xF; | ||||
|     const rm = cpu.r[opcode & 0xF]; | ||||
|     const rs = @truncate(u8, cpu.r[rs_idx]); | ||||
| 
 | ||||
|     const rm_idx = opcode & 0xF; | ||||
|     const rm = if (rm_idx == 0xF) cpu.fakePC() else cpu.r[rm_idx]; | ||||
| 
 | ||||
|     return switch (@truncate(u2, opcode >> 5)) { | ||||
|         0b00 => lsl(S, &cpu.cpsr, rm, rs), | ||||
|         0b01 => lsr(S, &cpu.cpsr, rm, rs), | ||||
|         0b10 => asr(S, &cpu.cpsr, rm, rs), | ||||
|         0b11 => ror(S, &cpu.cpsr, rm, rs), | ||||
|         0b00 => logicalLeft(S, &cpu.cpsr, rm, rs), | ||||
|         0b01 => logicalRight(S, &cpu.cpsr, rm, rs), | ||||
|         0b10 => arithmeticRight(S, &cpu.cpsr, rm, rs), | ||||
|         0b11 => rotateRight(S, &cpu.cpsr, rm, rs), | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| pub fn immediate(comptime S: bool, cpu: *Arm7tdmi, opcode: u32) u32 { | ||||
| pub fn immShift(comptime S: bool, cpu: *Arm7tdmi, opcode: u32) u32 { | ||||
|     const amount = @truncate(u8, opcode >> 7 & 0x1F); | ||||
|     const rm = cpu.r[opcode & 0xF]; | ||||
| 
 | ||||
|     const rm_idx = opcode & 0xF; | ||||
|     const rm = if (rm_idx == 0xF) cpu.fakePC() else cpu.r[rm_idx]; | ||||
| 
 | ||||
|     var result: u32 = undefined; | ||||
|     if (amount == 0) { | ||||
| @@ -58,17 +64,17 @@ pub fn immediate(comptime S: bool, cpu: *Arm7tdmi, opcode: u32) u32 { | ||||
|         } | ||||
|     } else { | ||||
|         switch (@truncate(u2, opcode >> 5)) { | ||||
|             0b00 => result = lsl(S, &cpu.cpsr, rm, amount), | ||||
|             0b01 => result = lsr(S, &cpu.cpsr, rm, amount), | ||||
|             0b10 => result = asr(S, &cpu.cpsr, rm, amount), | ||||
|             0b11 => result = ror(S, &cpu.cpsr, rm, amount), | ||||
|             0b00 => result = logicalLeft(S, &cpu.cpsr, rm, amount), | ||||
|             0b01 => result = logicalRight(S, &cpu.cpsr, rm, amount), | ||||
|             0b10 => result = arithmeticRight(S, &cpu.cpsr, rm, amount), | ||||
|             0b11 => result = rotateRight(S, &cpu.cpsr, rm, amount), | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| pub fn lsl(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u8) u32 { | ||||
| pub fn logicalLeft(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u8) u32 { | ||||
|     const amount = @truncate(u5, total_amount); | ||||
|     const bit_count: u8 = @typeInfo(u32).Int.bits; | ||||
| 
 | ||||
| @@ -95,7 +101,7 @@ pub fn lsl(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u8) u32 { | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| pub fn lsr(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u32) u32 { | ||||
| pub fn logicalRight(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u32) u32 { | ||||
|     const amount = @truncate(u5, total_amount); | ||||
|     const bit_count: u8 = @typeInfo(u32).Int.bits; | ||||
| 
 | ||||
| @@ -119,7 +125,7 @@ pub fn lsr(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u32) u32 { | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| pub fn asr(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u8) u32 { | ||||
| pub fn arithmeticRight(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u8) u32 { | ||||
|     const amount = @truncate(u5, total_amount); | ||||
|     const bit_count: u8 = @typeInfo(u32).Int.bits; | ||||
| 
 | ||||
| @@ -136,7 +142,7 @@ pub fn asr(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u8) u32 { | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| pub fn ror(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u8) u32 { | ||||
| pub fn rotateRight(comptime S: bool, cpsr: *CPSR, rm: u32, total_amount: u8) u32 { | ||||
|     const result = rotr(u32, rm, total_amount); | ||||
| 
 | ||||
|     if (S and total_amount != 0) { | ||||
							
								
								
									
										118
									
								
								src/cpu/thumb/alu.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								src/cpu/thumb/alu.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").ThumbInstrFn; | ||||
|  | ||||
| const adc = @import("../arm/data_processing.zig").adc; | ||||
| const sbc = @import("../arm/data_processing.zig").sbc; | ||||
| const sub = @import("../arm/data_processing.zig").sub; | ||||
| const cmp = @import("../arm/data_processing.zig").cmp; | ||||
| const cmn = @import("../arm/data_processing.zig").cmn; | ||||
| const setTestOpFlags = @import("../arm/data_processing.zig").setTestOpFlags; | ||||
| const setLogicOpFlags = @import("../arm/data_processing.zig").setLogicOpFlags; | ||||
|  | ||||
| const logicalLeft = @import("../barrel_shifter.zig").logicalLeft; | ||||
| const logicalRight = @import("../barrel_shifter.zig").logicalRight; | ||||
| const arithmeticRight = @import("../barrel_shifter.zig").arithmeticRight; | ||||
| const rotateRight = @import("../barrel_shifter.zig").rotateRight; | ||||
|  | ||||
| pub fn format4(comptime op: u4) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const rs = opcode >> 3 & 0x7; | ||||
|             const rd = opcode & 0x7; | ||||
|             const carry = @boolToInt(cpu.cpsr.c.read()); | ||||
|  | ||||
|             switch (op) { | ||||
|                 0x0 => { | ||||
|                     // AND | ||||
|                     const result = cpu.r[rd] & cpu.r[rs]; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|                 0x1 => { | ||||
|                     // EOR | ||||
|                     const result = cpu.r[rd] ^ cpu.r[rs]; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|                 0x2 => { | ||||
|                     // LSL | ||||
|                     const result = logicalLeft(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs])); | ||||
|                     cpu.r[rd] = result; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|                 0x3 => { | ||||
|                     // LSR | ||||
|                     const result = logicalRight(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs])); | ||||
|                     cpu.r[rd] = result; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|                 0x4 => { | ||||
|                     // ASR | ||||
|                     const result = arithmeticRight(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs])); | ||||
|                     cpu.r[rd] = result; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|                 0x5 => { | ||||
|                     // ADC | ||||
|                     cpu.r[rd] = adc(true, cpu, cpu.r[rd], cpu.r[rs], carry); | ||||
|                 }, | ||||
|                 0x6 => { | ||||
|                     // SBC | ||||
|                     cpu.r[rd] = sbc(true, cpu, cpu.r[rd], cpu.r[rs], carry); | ||||
|                 }, | ||||
|                 0x7 => { | ||||
|                     // ROR | ||||
|                     const result = rotateRight(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs])); | ||||
|                     cpu.r[rd] = result; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|                 0x8 => { | ||||
|                     // TST | ||||
|                     const result = cpu.r[rd] & cpu.r[rs]; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|                 0x9 => { | ||||
|                     // NEG | ||||
|                     cpu.r[rd] = sub(true, cpu, 0, cpu.r[rs]); | ||||
|                 }, | ||||
|                 0xA => { | ||||
|                     // CMP | ||||
|                     cmp(cpu, cpu.r[rd], cpu.r[rs]); | ||||
|                 }, | ||||
|                 0xB => { | ||||
|                     // CMN | ||||
|                     cmn(cpu, cpu.r[rd], cpu.r[rs]); | ||||
|                 }, | ||||
|                 0xC => { | ||||
|                     // ORR | ||||
|                     const result = cpu.r[rd] | cpu.r[rs]; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|                 0xD => { | ||||
|                     // MUL | ||||
|                     const temp = @as(u64, cpu.r[rs]) * @as(u64, cpu.r[rd]); | ||||
|                     const result = @truncate(u32, temp); | ||||
|                     cpu.r[rd] = result; | ||||
|  | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|                     // V is unaffected, assuming similar behaviour to ARMv4 MUL C is undefined | ||||
|                 }, | ||||
|                 0xE => { | ||||
|                     // BIC | ||||
|                     const result = cpu.r[rd] & ~cpu.r[rs]; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|                 0xF => { | ||||
|                     // MVN | ||||
|                     const result = ~cpu.r[rs]; | ||||
|                     cpu.r[rd] = result; | ||||
|                     setLogicOpFlags(true, cpu, result); | ||||
|                 }, | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").thumb.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ThumbInstrFn; | ||||
| 
 | ||||
| pub fn fmt14(comptime L: bool, comptime R: bool) InstrFn { | ||||
| pub fn format14(comptime L: bool, comptime R: bool) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, bus: *Bus, opcode: u16) void { | ||||
|             const count = @boolToInt(R) + countRlist(opcode); | ||||
| @@ -33,8 +33,7 @@ pub fn fmt14(comptime L: bool, comptime R: bool) InstrFn { | ||||
|             if (R) { | ||||
|                 if (L) { | ||||
|                     const value = bus.read(u32, address); | ||||
|                     cpu.r[15] = value & ~@as(u32, 1); | ||||
|                     cpu.pipe.reload(cpu); | ||||
|                     cpu.r[15] = value & 0xFFFF_FFFE; | ||||
|                 } else { | ||||
|                     bus.write(u32, address, cpu.r[14]); | ||||
|                 } | ||||
| @@ -46,20 +45,14 @@ pub fn fmt14(comptime L: bool, comptime R: bool) InstrFn { | ||||
|     }.inner; | ||||
| } | ||||
| 
 | ||||
| pub fn fmt15(comptime L: bool, comptime rb: u3) InstrFn { | ||||
| pub fn format15(comptime L: bool, comptime rb: u3) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, bus: *Bus, opcode: u16) void { | ||||
|             var address = cpu.r[rb]; | ||||
|             const end_address = cpu.r[rb] + 4 * countRlist(opcode); | ||||
| 
 | ||||
|             if (opcode & 0xFF == 0) { | ||||
|                 if (L) { | ||||
|                     cpu.r[15] = bus.read(u32, address); | ||||
|                     cpu.pipe.reload(cpu); | ||||
|                 } else { | ||||
|                     bus.write(u32, address, cpu.r[15] + 2); | ||||
|                 } | ||||
| 
 | ||||
|                 if (L) cpu.r[15] = bus.read(u32, address) else bus.write(u32, address, cpu.r[15] + 4); | ||||
|                 cpu.r[rb] += 0x40; | ||||
|                 return; | ||||
|             } | ||||
							
								
								
									
										54
									
								
								src/cpu/thumb/branch.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								src/cpu/thumb/branch.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").ThumbInstrFn; | ||||
|  | ||||
| const checkCond = @import("../../cpu.zig").checkCond; | ||||
| const sext = @import("../../util.zig").sext; | ||||
|  | ||||
| pub fn format16(comptime cond: u4) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             // B | ||||
|             const offset = sext(8, opcode & 0xFF) << 1; | ||||
|  | ||||
|             const should_execute = switch (cond) { | ||||
|                 0xE, 0xF => cpu.panic("[CPU/THUMB.16] Undefined conditional branch with condition {}", .{cond}), | ||||
|                 else => checkCond(cpu.cpsr, cond), | ||||
|             }; | ||||
|  | ||||
|             if (should_execute) { | ||||
|                 cpu.r[15] = (cpu.r[15] + 2) +% offset; | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn format18() InstrFn { | ||||
|     return struct { | ||||
|         // B but conditional | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const offset = sext(11, opcode & 0x7FF) << 1; | ||||
|             cpu.r[15] = (cpu.r[15] + 2) +% offset; | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn format19(comptime is_low: bool) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             // BL | ||||
|             const offset = opcode & 0x7FF; | ||||
|  | ||||
|             if (is_low) { | ||||
|                 // Instruction 2 | ||||
|                 const old_pc = cpu.r[15]; | ||||
|  | ||||
|                 cpu.r[15] = cpu.r[14] +% (offset << 1); | ||||
|                 cpu.r[14] = old_pc | 1; | ||||
|             } else { | ||||
|                 // Instruction 1 | ||||
|                 cpu.r[14] = (cpu.r[15] + 2) +% (sext(11, offset) << 12); | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
							
								
								
									
										122
									
								
								src/cpu/thumb/data_processing.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										122
									
								
								src/cpu/thumb/data_processing.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,122 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").ThumbInstrFn; | ||||
| const shifter = @import("../barrel_shifter.zig"); | ||||
|  | ||||
| const add = @import("../arm/data_processing.zig").add; | ||||
| const sub = @import("../arm/data_processing.zig").sub; | ||||
| const cmp = @import("../arm/data_processing.zig").cmp; | ||||
| const setLogicOpFlags = @import("../arm/data_processing.zig").setLogicOpFlags; | ||||
|  | ||||
| const log = std.log.scoped(.Thumb1); | ||||
|  | ||||
| pub fn format1(comptime op: u2, comptime offset: u5) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const rs = opcode >> 3 & 0x7; | ||||
|             const rd = opcode & 0x7; | ||||
|  | ||||
|             const result = switch (op) { | ||||
|                 0b00 => blk: { | ||||
|                     // LSL | ||||
|                     if (offset == 0) { | ||||
|                         break :blk cpu.r[rs]; | ||||
|                     } else { | ||||
|                         break :blk shifter.logicalLeft(true, &cpu.cpsr, cpu.r[rs], offset); | ||||
|                     } | ||||
|                 }, | ||||
|                 0b01 => blk: { | ||||
|                     // LSR | ||||
|                     if (offset == 0) { | ||||
|                         cpu.cpsr.c.write(cpu.r[rs] >> 31 & 1 == 1); | ||||
|                         break :blk @as(u32, 0); | ||||
|                     } else { | ||||
|                         break :blk shifter.logicalRight(true, &cpu.cpsr, cpu.r[rs], offset); | ||||
|                     } | ||||
|                 }, | ||||
|                 0b10 => blk: { | ||||
|                     // ASR | ||||
|                     if (offset == 0) { | ||||
|                         cpu.cpsr.c.write(cpu.r[rs] >> 31 & 1 == 1); | ||||
|                         break :blk @bitCast(u32, @bitCast(i32, cpu.r[rs]) >> 31); | ||||
|                     } else { | ||||
|                         break :blk shifter.arithmeticRight(true, &cpu.cpsr, cpu.r[rs], offset); | ||||
|                     } | ||||
|                 }, | ||||
|                 else => cpu.panic("[CPU/THUMB.1] 0b{b:0>2} is not a valid op", .{op}), | ||||
|             }; | ||||
|  | ||||
|             // Equivalent to an ARM MOVS | ||||
|             cpu.r[rd] = result; | ||||
|             setLogicOpFlags(true, cpu, result); | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn format2(comptime I: bool, is_sub: bool, rn: u3) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const rs = opcode >> 3 & 0x7; | ||||
|             const rd = @truncate(u3, opcode); | ||||
|  | ||||
|             if (is_sub) { | ||||
|                 // SUB | ||||
|                 cpu.r[rd] = if (I) blk: { | ||||
|                     break :blk sub(true, cpu, cpu.r[rs], rn); | ||||
|                 } else blk: { | ||||
|                     break :blk sub(true, cpu, cpu.r[rs], cpu.r[rn]); | ||||
|                 }; | ||||
|             } else { | ||||
|                 // ADD | ||||
|                 cpu.r[rd] = if (I) blk: { | ||||
|                     break :blk add(true, cpu, cpu.r[rs], rn); | ||||
|                 } else blk: { | ||||
|                     break :blk add(true, cpu, cpu.r[rs], cpu.r[rn]); | ||||
|                 }; | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn format3(comptime op: u2, comptime rd: u3) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const offset = @truncate(u8, opcode); | ||||
|  | ||||
|             switch (op) { | ||||
|                 0b00 => { | ||||
|                     // MOV | ||||
|                     cpu.r[rd] = offset; | ||||
|                     setLogicOpFlags(true, cpu, offset); | ||||
|                 }, | ||||
|                 0b01 => cmp(cpu, cpu.r[rd], offset), // CMP | ||||
|                 0b10 => cpu.r[rd] = add(true, cpu, cpu.r[rd], offset), // ADD | ||||
|                 0b11 => cpu.r[rd] = sub(true, cpu, cpu.r[rd], offset), // SUB | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn format12(comptime isSP: bool, comptime rd: u3) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             // ADD | ||||
|             const left = if (isSP) cpu.r[13] else (cpu.r[15] + 2) & 0xFFFF_FFFD; | ||||
|             const right = (opcode & 0xFF) << 2; | ||||
|             const result = left + right; | ||||
|             cpu.r[rd] = result; | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| pub fn format13(comptime S: bool) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             // ADD | ||||
|             const offset = (opcode & 0x7F) << 2; | ||||
|             cpu.r[13] = if (S) cpu.r[13] - offset else cpu.r[13] + offset; | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
| @@ -1,30 +1,31 @@ | ||||
| const std = @import("std"); | ||||
| 
 | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").thumb.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ThumbInstrFn; | ||||
| 
 | ||||
| const rotr = @import("../../../util.zig").rotr; | ||||
| const sext = @import("../../../util.zig").sext; | ||||
| const rotr = @import("../../util.zig").rotr; | ||||
| 
 | ||||
| pub fn fmt6(comptime rd: u3) InstrFn { | ||||
| pub fn format6(comptime rd: u3) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, bus: *Bus, opcode: u16) void { | ||||
|             // LDR | ||||
|             const offset = (opcode & 0xFF) << 2; | ||||
| 
 | ||||
|             // Bit 1 of the PC intentionally ignored | ||||
|             cpu.r[rd] = bus.read(u32, (cpu.r[15] & ~@as(u32, 2)) + offset); | ||||
|             cpu.r[rd] = bus.read(u32, (cpu.r[15] + 2 & 0xFFFF_FFFD) + offset); | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
| 
 | ||||
| pub fn fmt78(comptime op: u2, comptime T: bool) InstrFn { | ||||
| const sext = @import("../../util.zig").sext; | ||||
| 
 | ||||
| pub fn format78(comptime op: u2, comptime T: bool) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, bus: *Bus, opcode: u16) void { | ||||
|             const ro = opcode >> 6 & 0x7; | ||||
|             const rb = opcode >> 3 & 0x7; | ||||
|             const rd = opcode & 0x7; | ||||
| 
 | ||||
|             const address = cpu.r[rb] +% cpu.r[ro]; | ||||
|             const address = cpu.r[rb] + cpu.r[ro]; | ||||
| 
 | ||||
|             if (T) { | ||||
|                 // Format 8 | ||||
| @@ -35,7 +36,7 @@ pub fn fmt78(comptime op: u2, comptime T: bool) InstrFn { | ||||
|                     }, | ||||
|                     0b01 => { | ||||
|                         // LDSB | ||||
|                         cpu.r[rd] = sext(u32, u8, bus.read(u8, address)); | ||||
|                         cpu.r[rd] = sext(8, bus.read(u8, address)); | ||||
|                     }, | ||||
|                     0b10 => { | ||||
|                         // LDRH | ||||
| @@ -44,8 +45,11 @@ pub fn fmt78(comptime op: u2, comptime T: bool) InstrFn { | ||||
|                     }, | ||||
|                     0b11 => { | ||||
|                         // LDRSH | ||||
|                         const value = bus.read(u16, address); | ||||
|                         cpu.r[rd] = if (address & 1 == 1) sext(u32, u8, @truncate(u8, value >> 8)) else sext(u32, u16, value); | ||||
|                         cpu.r[rd] = if (address & 1 == 1) blk: { | ||||
|                             break :blk sext(8, bus.read(u8, address)); | ||||
|                         } else blk: { | ||||
|                             break :blk sext(16, bus.read(u16, address)); | ||||
|                         }; | ||||
|                     }, | ||||
|                 } | ||||
|             } else { | ||||
| @@ -74,7 +78,7 @@ pub fn fmt78(comptime op: u2, comptime T: bool) InstrFn { | ||||
|     }.inner; | ||||
| } | ||||
| 
 | ||||
| pub fn fmt9(comptime B: bool, comptime L: bool, comptime offset: u5) InstrFn { | ||||
| pub fn format9(comptime B: bool, comptime L: bool, comptime offset: u5) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, bus: *Bus, opcode: u16) void { | ||||
|             const rb = opcode >> 3 & 0x7; | ||||
| @@ -106,7 +110,7 @@ pub fn fmt9(comptime B: bool, comptime L: bool, comptime offset: u5) InstrFn { | ||||
|     }.inner; | ||||
| } | ||||
| 
 | ||||
| pub fn fmt10(comptime L: bool, comptime offset: u5) InstrFn { | ||||
| pub fn format10(comptime L: bool, comptime offset: u5) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, bus: *Bus, opcode: u16) void { | ||||
|             const rb = opcode >> 3 & 0x7; | ||||
| @@ -126,7 +130,7 @@ pub fn fmt10(comptime L: bool, comptime offset: u5) InstrFn { | ||||
|     }.inner; | ||||
| } | ||||
| 
 | ||||
| pub fn fmt11(comptime L: bool, comptime rd: u3) InstrFn { | ||||
| pub fn format11(comptime L: bool, comptime rd: u3) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, bus: *Bus, opcode: u16) void { | ||||
|             const offset = (opcode & 0xFF) << 2; | ||||
							
								
								
									
										36
									
								
								src/cpu/thumb/processing_branch.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								src/cpu/thumb/processing_branch.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").ThumbInstrFn; | ||||
|  | ||||
| const cmp = @import("../arm/data_processing.zig").cmp; | ||||
| const add = @import("../arm/data_processing.zig").add; | ||||
|  | ||||
| pub fn format5(comptime op: u2, comptime h1: u1, comptime h2: u1) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u16) void { | ||||
|             const src_idx = @as(u4, h2) << 3 | (opcode >> 3 & 0x7); | ||||
|             const dst_idx = @as(u4, h1) << 3 | (opcode & 0x7); | ||||
|  | ||||
|             const src = if (src_idx == 0xF) (cpu.r[src_idx] + 2) & 0xFFFF_FFFE else cpu.r[src_idx]; | ||||
|             const dst = if (dst_idx == 0xF) (cpu.r[dst_idx] + 2) & 0xFFFF_FFFE else cpu.r[dst_idx]; | ||||
|  | ||||
|             switch (op) { | ||||
|                 0b00 => { | ||||
|                     // ADD | ||||
|                     const sum = add(false, cpu, dst, src); | ||||
|                     cpu.r[dst_idx] = if (dst_idx == 0xF) sum & 0xFFFF_FFFE else sum; | ||||
|                 }, | ||||
|                 0b01 => cmp(cpu, dst, src), // CMP | ||||
|                 0b10 => { | ||||
|                     // MOV | ||||
|                     cpu.r[dst_idx] = if (dst_idx == 0xF) src & 0xFFFF_FFFE else src; | ||||
|                 }, | ||||
|                 0b11 => { | ||||
|                     // BX | ||||
|                     cpu.cpsr.t.write(src & 1 == 1); | ||||
|                     cpu.r[15] = src & 0xFFFF_FFFE; | ||||
|                 }, | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
| @@ -1,12 +1,12 @@ | ||||
| const Bus = @import("../../Bus.zig"); | ||||
| const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; | ||||
| const InstrFn = @import("../../cpu.zig").thumb.InstrFn; | ||||
| const InstrFn = @import("../../cpu.zig").ThumbInstrFn; | ||||
| 
 | ||||
| pub fn fmt17() InstrFn { | ||||
| pub fn thumbSoftwareInterrupt() InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, _: u16) void { | ||||
|             // Copy Values from Current Mode | ||||
|             const ret_addr = cpu.r[15] - 2; | ||||
|             const r15 = cpu.r[15]; | ||||
|             const cpsr = cpu.cpsr.raw; | ||||
| 
 | ||||
|             // Switch Mode | ||||
| @@ -14,10 +14,9 @@ pub fn fmt17() InstrFn { | ||||
|             cpu.cpsr.t.write(false); // Force ARM Mode | ||||
|             cpu.cpsr.i.write(true); // Disable normal interrupts | ||||
| 
 | ||||
|             cpu.r[14] = ret_addr; // Resume Execution | ||||
|             cpu.r[14] = r15; // Resume Execution | ||||
|             cpu.spsr.raw = cpsr; // Previous mode CPSR | ||||
|             cpu.r[15] = 0x0000_0008; | ||||
|             cpu.pipe.reload(cpu); | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
							
								
								
									
										160
									
								
								src/emu.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										160
									
								
								src/emu.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,160 @@ | ||||
| const std = @import("std"); | ||||
|  | ||||
| const Bus = @import("Bus.zig"); | ||||
| const Scheduler = @import("scheduler.zig").Scheduler; | ||||
| const Arm7tdmi = @import("cpu.zig").Arm7tdmi; | ||||
| const FpsAverage = @import("util.zig").FpsAverage; | ||||
|  | ||||
| const Timer = std.time.Timer; | ||||
| const Thread = std.Thread; | ||||
| const Atomic = std.atomic.Atomic; | ||||
|  | ||||
| // 228 Lines which consist of 308 dots (which are 4 cycles long) | ||||
| const cycles_per_frame: u64 = 228 * (308 * 4); //280896 | ||||
| const clock_rate: u64 = 1 << 24; // 16.78MHz | ||||
|  | ||||
| // TODO: Don't truncate this, be more accurate w/ timing | ||||
| // 59.6046447754ns (truncated to just 59ns) | ||||
| const clock_period: u64 = std.time.ns_per_s / clock_rate; | ||||
| const frame_period = (clock_period * cycles_per_frame); | ||||
|  | ||||
| // 59.7275005696Hz | ||||
| pub const frame_rate = @intToFloat(f64, std.time.ns_per_s) / | ||||
|     ((@intToFloat(f64, std.time.ns_per_s) / @intToFloat(f64, clock_rate)) * @intToFloat(f64, cycles_per_frame)); | ||||
|  | ||||
| const log = std.log.scoped(.Emulation); | ||||
|  | ||||
| const RunKind = enum { | ||||
|     Unlimited, | ||||
|     UnlimitedFPS, | ||||
|     Limited, | ||||
|     LimitedFPS, | ||||
|     LimitedBusy, | ||||
| }; | ||||
|  | ||||
| pub fn run(kind: RunKind, quit: *Atomic(bool), fps: *FpsAverage, sched: *Scheduler, cpu: *Arm7tdmi) void { | ||||
|     switch (kind) { | ||||
|         .Unlimited => runUnsync(quit, sched, cpu), | ||||
|         .Limited => runSync(quit, sched, cpu), | ||||
|         .UnlimitedFPS => runUnsyncFps(quit, fps, sched, cpu), | ||||
|         .LimitedFPS => runSyncFps(quit, fps, sched, cpu), | ||||
|         .LimitedBusy => runBusyLoop(quit, sched, cpu), | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn runFrame(sched: *Scheduler, cpu: *Arm7tdmi) void { | ||||
|     const frame_end = sched.tick + cycles_per_frame; | ||||
|  | ||||
|     while (sched.tick < frame_end) { | ||||
|         if (cpu.bus.io.haltcnt == .Halt) sched.tick += 1; | ||||
|         if (cpu.bus.io.haltcnt == .Execute) cpu.step(); | ||||
|         cpu.bus.handleDMATransfers(); | ||||
|  | ||||
|         while (sched.tick >= sched.nextTimestamp()) { | ||||
|             sched.handleEvent(cpu); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn runUnsync(quit: *Atomic(bool), sched: *Scheduler, cpu: *Arm7tdmi) void { | ||||
|     log.info("Unsynchronized EmuThread has begun", .{}); | ||||
|     while (!quit.load(.Unordered)) runFrame(sched, cpu); | ||||
| } | ||||
|  | ||||
| pub fn runSync(quit: *Atomic(bool), sched: *Scheduler, cpu: *Arm7tdmi) void { | ||||
|     log.info("Synchronized EmuThread has begun", .{}); | ||||
|     var timer = Timer.start() catch unreachable; | ||||
|     var wake_time: u64 = frame_period; | ||||
|  | ||||
|     while (!quit.load(.Unordered)) { | ||||
|         runFrame(sched, cpu); | ||||
|  | ||||
|         // Put the Thread to Sleep + Backup Spin Loop | ||||
|         // This saves on resource usage when frame limiting | ||||
|         sleep(&timer, &wake_time); | ||||
|  | ||||
|         // Update to the new wake time | ||||
|         wake_time += frame_period; | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn runUnsyncFps(quit: *Atomic(bool), fps: *FpsAverage, sched: *Scheduler, cpu: *Arm7tdmi) void { | ||||
|     log.info("Unsynchronized EmuThread with FPS Tracking has begun", .{}); | ||||
|     var fps_timer = Timer.start() catch unreachable; | ||||
|  | ||||
|     while (!quit.load(.Unordered)) { | ||||
|         runFrame(sched, cpu); | ||||
|         fps.add(fps_timer.lap()); | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn runSyncFps(quit: *Atomic(bool), fps: *FpsAverage, sched: *Scheduler, cpu: *Arm7tdmi) void { | ||||
|     log.info("Synchronized EmuThread has begun", .{}); | ||||
|     var timer = Timer.start() catch unreachable; | ||||
|     var fps_timer = Timer.start() catch unreachable; | ||||
|     var wake_time: u64 = frame_period; | ||||
|  | ||||
|     while (!quit.load(.Unordered)) { | ||||
|         runFrame(sched, cpu); | ||||
|  | ||||
|         // Put the Thread to Sleep + Backup Spin Loop | ||||
|         // This saves on resource usage when frame limiting | ||||
|         sleep(&timer, &wake_time); | ||||
|  | ||||
|         // Determine FPS | ||||
|         fps.add(fps_timer.lap()); | ||||
|  | ||||
|         // Update to the new wake time | ||||
|         wake_time += frame_period; | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn runBusyLoop(quit: *Atomic(bool), sched: *Scheduler, cpu: *Arm7tdmi) void { | ||||
|     log.info("Run EmuThread with spin-loop sync", .{}); | ||||
|     var timer = Timer.start() catch unreachable; | ||||
|     var wake_time: u64 = frame_period; | ||||
|  | ||||
|     while (!quit.load(.Unordered)) { | ||||
|         runFrame(sched, cpu); | ||||
|         spinLoop(&timer, wake_time); | ||||
|  | ||||
|         // Update to the new wake time | ||||
|         wake_time += frame_period; | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn sleep(timer: *Timer, wake_time: *u64) void { | ||||
|     // const step = std.time.ns_per_ms * 10; // 10ms | ||||
|     const timestamp = timer.read(); | ||||
|  | ||||
|     // ns_late is non zero if we are late. | ||||
|     const ns_late = timestamp -| wake_time.*; | ||||
|  | ||||
|     // If we're more than a frame late, skip the rest of this loop | ||||
|     // Recalculate what our new wake time should be so that we can | ||||
|     // get "back on track" | ||||
|     if (ns_late > frame_period) { | ||||
|         wake_time.* = timestamp + frame_period; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     const sleep_for = frame_period - ns_late; | ||||
|  | ||||
|     // // Employ several sleep calls in periods of 10ms | ||||
|     // // By doing this the behaviour should average out to be | ||||
|     // // more consistent | ||||
|     // const loop_count = sleep_for / step; // How many groups of 10ms | ||||
|  | ||||
|     // var i: usize = 0; | ||||
|     // while (i < loop_count) : (i += 1) std.time.sleep(step); | ||||
|  | ||||
|     std.time.sleep(sleep_for); | ||||
|  | ||||
|     // Spin to make up the difference if there is a need | ||||
|     // Make sure that we're using the old wake time and not the onne we recalculated | ||||
|     spinLoop(timer, wake_time.*); | ||||
| } | ||||
|  | ||||
| fn spinLoop(timer: *Timer, wake_time: u64) void { | ||||
|     while (true) if (timer.read() > wake_time) break; | ||||
| } | ||||
							
								
								
									
										313
									
								
								src/main.zig
									
									
									
									
									
								
							
							
						
						
									
										313
									
								
								src/main.zig
									
									
									
									
									
								
							| @@ -1,160 +1,227 @@ | ||||
| const std = @import("std"); | ||||
| const builtin = @import("builtin"); | ||||
| const known_folders = @import("known_folders"); | ||||
| const SDL = @import("sdl2"); | ||||
| const clap = @import("clap"); | ||||
| const known_folders = @import("known_folders"); | ||||
|  | ||||
| const config = @import("config.zig"); | ||||
| const emu = @import("emu.zig"); | ||||
| const Bus = @import("Bus.zig"); | ||||
| const Arm7tdmi = @import("cpu.zig").Arm7tdmi; | ||||
| const Scheduler = @import("scheduler.zig").Scheduler; | ||||
| const FpsAverage = @import("util.zig").FpsAverage; | ||||
|  | ||||
| const Gui = @import("platform.zig").Gui; | ||||
| const Bus = @import("core/Bus.zig"); | ||||
| const Arm7tdmi = @import("core/cpu.zig").Arm7tdmi; | ||||
| const Scheduler = @import("core/scheduler.zig").Scheduler; | ||||
| const FilePaths = @import("util.zig").FilePaths; | ||||
| const Timer = std.time.Timer; | ||||
| const Thread = std.Thread; | ||||
| const Atomic = std.atomic.Atomic; | ||||
| const File = std.fs.File; | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
| const log = std.log.scoped(.Cli); | ||||
| const width = @import("core/ppu.zig").width; | ||||
| const height = @import("core/ppu.zig").height; | ||||
| pub const log_level = if (builtin.mode != .Debug) .info else std.log.default_level; | ||||
| const window_scale = 3; | ||||
| const gba_width = @import("ppu.zig").width; | ||||
| const gba_height = @import("ppu.zig").height; | ||||
| const framebuf_pitch = @import("ppu.zig").framebuf_pitch; | ||||
| const expected_rate = @import("emu.zig").frame_rate; | ||||
|  | ||||
| // CLI Arguments + Help Text | ||||
| const params = clap.parseParamsComptime( | ||||
|     \\-h, --help            Display this help and exit. | ||||
|     \\-s, --skip            Skip BIOS. | ||||
|     \\-b, --bios <str>      Optional path to a GBA BIOS ROM. | ||||
|     \\<str>                 Path to the GBA GamePak ROM. | ||||
|     \\ | ||||
| ); | ||||
| pub const enable_logging: bool = false; | ||||
| const is_binary: bool = false; | ||||
| const log = std.log.scoped(.GUI); | ||||
|  | ||||
| pub fn main() void { | ||||
|     // Main Allocator for ZBA | ||||
| const correctTitle = @import("util.zig").correctTitle; | ||||
|  | ||||
| pub fn main() anyerror!void { | ||||
|     // Allocator for Emulator + CLI | ||||
|     var gpa = std.heap.GeneralPurposeAllocator(.{}){}; | ||||
|     const alloc = gpa.allocator(); | ||||
|     defer std.debug.assert(!gpa.deinit()); | ||||
|  | ||||
|     const allocator = gpa.allocator(); | ||||
|     // CLI Arguments | ||||
|     const params = comptime clap.parseParamsComptime( | ||||
|         \\-h, --help            Display this help and exit. | ||||
|         \\-b, --bios <str>      Optional path to a GBA BIOS ROM. | ||||
|         \\<str>                 Path to the GBA GamePak ROM | ||||
|         \\ | ||||
|     ); | ||||
|  | ||||
|     // Determine the Data Directory (stores saves) | ||||
|     const data_path = blk: { | ||||
|         const result = known_folders.getPath(allocator, .data); | ||||
|         const option = result catch |e| exitln("interrupted while determining the data folder: {}", .{e}); | ||||
|         const path = option orelse exitln("no valid data folder found", .{}); | ||||
|         ensureDataDirsExist(path) catch |e| exitln("failed to create folders under \"{s}\": {}", .{ path, e }); | ||||
|     var res = try clap.parse(clap.Help, ¶ms, clap.parsers.default, .{}); | ||||
|     defer res.deinit(); | ||||
|  | ||||
|         break :blk path; | ||||
|     const stderr = std.io.getStdErr(); | ||||
|     defer stderr.close(); | ||||
|  | ||||
|     if (res.args.help) return clap.help(stderr.writer(), clap.Help, ¶ms, .{}); | ||||
|     const bios_path: ?[]const u8 = if (res.args.bios) |p| p else null; | ||||
|  | ||||
|     const rom_path = switch (res.positionals.len) { | ||||
|         1 => res.positionals[0], | ||||
|         0 => { | ||||
|             try stderr.writeAll("ZBA requires a positional path to a GamePak ROM.\n"); | ||||
|             return CliError.InsufficientOptions; | ||||
|         }, | ||||
|         else => { | ||||
|             try stderr.writeAll("ZBA received too many arguments.\n"); | ||||
|             return CliError.UnneededOptions; | ||||
|         }, | ||||
|     }; | ||||
|     defer allocator.free(data_path); | ||||
|  | ||||
|     // Determine the Config Directory | ||||
|     const config_path = blk: { | ||||
|         const result = known_folders.getPath(allocator, .roaming_configuration); | ||||
|         const option = result catch |e| exitln("interreupted while determining the config folder: {}", .{e}); | ||||
|         const path = option orelse exitln("no valid config folder found", .{}); | ||||
|         ensureConfigDirExists(path) catch |e| exitln("failed to create required folder \"{s}\": {}", .{ path, e }); | ||||
|     // Determine Save Directory | ||||
|     const save_path = try setupSavePath(alloc); | ||||
|     defer if (save_path) |path| alloc.free(path); | ||||
|     log.info("Save Path: {s}", .{save_path}); | ||||
|  | ||||
|         break :blk path; | ||||
|     }; | ||||
|     defer allocator.free(config_path); | ||||
|     // Initialize SDL | ||||
|     const status = SDL.SDL_Init(SDL.SDL_INIT_VIDEO | SDL.SDL_INIT_EVENTS | SDL.SDL_INIT_AUDIO | SDL.SDL_INIT_GAMECONTROLLER); | ||||
|     defer SDL.SDL_Quit(); | ||||
|     if (status < 0) sdlPanic(); | ||||
|  | ||||
|     // Parse CLI | ||||
|     const result = clap.parse(clap.Help, ¶ms, clap.parsers.default, .{}) catch |e| exitln("failed to parse cli: {}", .{e}); | ||||
|     defer result.deinit(); | ||||
|     // Initialize SDL Audio | ||||
|     var have: SDL.SDL_AudioSpec = undefined; | ||||
|     var want = std.mem.zeroes(SDL.SDL_AudioSpec); | ||||
|     want.freq = 32768; | ||||
|     want.format = SDL.AUDIO_S8; | ||||
|     want.channels = 2; | ||||
|     want.samples = 0x200; | ||||
|     want.callback = null; | ||||
|  | ||||
|     // TODO: Move config file to XDG Config directory? | ||||
|     const cfg_file_path = configFilePath(allocator, config_path) catch |e| exitln("failed to ready config file for access: {}", .{e}); | ||||
|     defer allocator.free(cfg_file_path); | ||||
|     const audio_dev = SDL.SDL_OpenAudioDevice(null, 0, &want, &have, 0); | ||||
|     defer SDL.SDL_CloseAudioDevice(audio_dev); | ||||
|     if (audio_dev == 0) sdlPanic(); | ||||
|  | ||||
|     config.load(allocator, cfg_file_path) catch |e| exitln("failed to load config file: {}", .{e}); | ||||
|     // Start Playback on the Audio evice | ||||
|     SDL.SDL_PauseAudioDevice(audio_dev, 0); | ||||
|  | ||||
|     const paths = handleArguments(allocator, data_path, &result) catch |e| exitln("failed to handle cli arguments: {}", .{e}); | ||||
|     defer if (paths.save) |path| allocator.free(path); | ||||
|     // Initialize Emulator | ||||
|     var scheduler = Scheduler.init(alloc); | ||||
|     defer scheduler.deinit(); | ||||
|  | ||||
|     const log_file = if (config.config().debug.cpu_trace) blk: { | ||||
|         break :blk std.fs.cwd().createFile("zba.log", .{}) catch |e| exitln("failed to create trace log file: {}", .{e}); | ||||
|     const paths = .{ .bios = bios_path, .rom = rom_path, .save = save_path }; | ||||
|     var cpu = try Arm7tdmi.init(alloc, &scheduler, paths); | ||||
|     cpu.bus.apu.attachAudioDevice(audio_dev); | ||||
|     cpu.fastBoot(); | ||||
|  | ||||
|     const log_file: ?File = if (enable_logging) blk: { | ||||
|         const file = try std.fs.cwd().createFile(if (is_binary) "zba.bin" else "zba.log", .{}); | ||||
|         cpu.useLogger(&file, is_binary); | ||||
|         break :blk file; | ||||
|     } else null; | ||||
|     defer if (log_file) |file| file.close(); | ||||
|  | ||||
|     // TODO: Take Emulator Init Code out of main.zig | ||||
|     var scheduler = Scheduler.init(allocator); | ||||
|     defer scheduler.deinit(); | ||||
|     // Init Atomics | ||||
|     var quit = Atomic(bool).init(false); | ||||
|     var emu_rate = FpsAverage.init(); | ||||
|  | ||||
|     var bus: Bus = undefined; | ||||
|     var cpu = Arm7tdmi.init(&scheduler, &bus, log_file); | ||||
|     // Create Emulator Thread | ||||
|     const emu_thread = try Thread.spawn(.{}, emu.run, .{ .LimitedFPS, &quit, &emu_rate, &scheduler, &cpu }); | ||||
|     defer emu_thread.join(); | ||||
|  | ||||
|     bus.init(allocator, &scheduler, &cpu, paths) catch |e| exitln("failed to init zba bus: {}", .{e}); | ||||
|     defer bus.deinit(); | ||||
|     const title = correctTitle(cpu.bus.pak.title); | ||||
|  | ||||
|     if (config.config().guest.skip_bios or result.args.skip or paths.bios == null) { | ||||
|         cpu.fastBoot(); | ||||
|     var title_buf: [0x20]u8 = std.mem.zeroes([0x20]u8); | ||||
|     const window_title = try std.fmt.bufPrint(&title_buf, "ZBA | {s}", .{title}); | ||||
|  | ||||
|     var window = SDL.SDL_CreateWindow( | ||||
|         window_title.ptr, | ||||
|         SDL.SDL_WINDOWPOS_CENTERED, | ||||
|         SDL.SDL_WINDOWPOS_CENTERED, | ||||
|         gba_width * window_scale, | ||||
|         gba_height * window_scale, | ||||
|         SDL.SDL_WINDOW_SHOWN, | ||||
|     ) orelse sdlPanic(); | ||||
|     defer SDL.SDL_DestroyWindow(window); | ||||
|  | ||||
|     const renderer = SDL.SDL_CreateRenderer(window, -1, SDL.SDL_RENDERER_ACCELERATED | SDL.SDL_RENDERER_PRESENTVSYNC) orelse sdlPanic(); | ||||
|     defer SDL.SDL_DestroyRenderer(renderer); | ||||
|  | ||||
|     const texture = SDL.SDL_CreateTexture(renderer, SDL.SDL_PIXELFORMAT_RGBA8888, SDL.SDL_TEXTUREACCESS_STREAMING, 240, 160) orelse sdlPanic(); | ||||
|     defer SDL.SDL_DestroyTexture(texture); | ||||
|  | ||||
|     // Init FPS Timer | ||||
|     var dyn_title_buf: [0x100]u8 = [_]u8{0x00} ** 0x100; | ||||
|  | ||||
|     emu_loop: while (true) { | ||||
|         var event: SDL.SDL_Event = undefined; | ||||
|         while (SDL.SDL_PollEvent(&event) != 0) { | ||||
|             // Pause Emulation Thread during Input Writing | ||||
|  | ||||
|             switch (event.type) { | ||||
|                 SDL.SDL_QUIT => break :emu_loop, | ||||
|                 SDL.SDL_KEYDOWN => { | ||||
|                     const io = &cpu.bus.io; | ||||
|                     const key_code = event.key.keysym.sym; | ||||
|  | ||||
|                     switch (key_code) { | ||||
|                         SDL.SDLK_UP => io.keyinput.up.unset(), | ||||
|                         SDL.SDLK_DOWN => io.keyinput.down.unset(), | ||||
|                         SDL.SDLK_LEFT => io.keyinput.left.unset(), | ||||
|                         SDL.SDLK_RIGHT => io.keyinput.right.unset(), | ||||
|                         SDL.SDLK_x => io.keyinput.a.unset(), | ||||
|                         SDL.SDLK_z => io.keyinput.b.unset(), | ||||
|                         SDL.SDLK_a => io.keyinput.shoulder_l.unset(), | ||||
|                         SDL.SDLK_s => io.keyinput.shoulder_r.unset(), | ||||
|                         SDL.SDLK_RETURN => io.keyinput.start.unset(), | ||||
|                         SDL.SDLK_RSHIFT => io.keyinput.select.unset(), | ||||
|                         else => {}, | ||||
|                     } | ||||
|                 }, | ||||
|                 SDL.SDL_KEYUP => { | ||||
|                     const io = &cpu.bus.io; | ||||
|                     const key_code = event.key.keysym.sym; | ||||
|  | ||||
|                     switch (key_code) { | ||||
|                         SDL.SDLK_UP => io.keyinput.up.set(), | ||||
|                         SDL.SDLK_DOWN => io.keyinput.down.set(), | ||||
|                         SDL.SDLK_LEFT => io.keyinput.left.set(), | ||||
|                         SDL.SDLK_RIGHT => io.keyinput.right.set(), | ||||
|                         SDL.SDLK_x => io.keyinput.a.set(), | ||||
|                         SDL.SDLK_z => io.keyinput.b.set(), | ||||
|                         SDL.SDLK_a => io.keyinput.shoulder_l.set(), | ||||
|                         SDL.SDLK_s => io.keyinput.shoulder_r.set(), | ||||
|                         SDL.SDLK_RETURN => io.keyinput.start.set(), | ||||
|                         SDL.SDLK_RSHIFT => io.keyinput.select.set(), | ||||
|                         else => {}, | ||||
|                     } | ||||
|                 }, | ||||
|                 else => {}, | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // FIXME: Is it OK just to copy the Emulator's Frame Buffer to SDL? | ||||
|         const buf_ptr = cpu.bus.ppu.framebuf.ptr; | ||||
|         _ = SDL.SDL_UpdateTexture(texture, null, buf_ptr, framebuf_pitch); | ||||
|         _ = SDL.SDL_RenderCopy(renderer, texture, null, null); | ||||
|         SDL.SDL_RenderPresent(renderer); | ||||
|  | ||||
|         const actual = emu_rate.calc(); | ||||
|         const dyn_title = std.fmt.bufPrint(&dyn_title_buf, "{s} [Emu: {d:0>3.2}fps, {d:0>3.2}%] ", .{ window_title, actual, actual * 100 / expected_rate }) catch unreachable; | ||||
|         SDL.SDL_SetWindowTitle(window, dyn_title.ptr); | ||||
|     } | ||||
|  | ||||
|     var gui = Gui.init(&bus.pak.title, &bus.apu, width, height) catch |e| exitln("failed to init gui: {}", .{e}); | ||||
|     defer gui.deinit(); | ||||
|  | ||||
|     gui.run(&cpu, &scheduler) catch |e| exitln("failed to run gui thread: {}", .{e}); | ||||
|     quit.store(true, .Unordered); // Terminate Emulator Thread | ||||
| } | ||||
|  | ||||
| pub fn handleArguments(allocator: Allocator, data_path: []const u8, result: *const clap.Result(clap.Help, ¶ms, clap.parsers.default)) !FilePaths { | ||||
|     const rom_path = romPath(result); | ||||
|     log.info("ROM path: {s}", .{rom_path}); | ||||
|  | ||||
|     const bios_path = result.args.bios; | ||||
|     if (bios_path) |path| log.info("BIOS path: {s}", .{path}) else log.warn("No BIOS provided", .{}); | ||||
|  | ||||
|     const save_path = try std.fs.path.join(allocator, &[_][]const u8{ data_path, "zba", "save" }); | ||||
|     log.info("Save path: {s}", .{save_path}); | ||||
|  | ||||
|     return .{ | ||||
|         .rom = rom_path, | ||||
|         .bios = bios_path, | ||||
|         .save = save_path, | ||||
|     }; | ||||
| fn sdlPanic() noreturn { | ||||
|     const str = @as(?[*:0]const u8, SDL.SDL_GetError()) orelse "unknown error"; | ||||
|     @panic(std.mem.sliceTo(str, 0)); | ||||
| } | ||||
|  | ||||
| fn configFilePath(allocator: Allocator, config_path: []const u8) ![]const u8 { | ||||
|     const path = try std.fs.path.join(allocator, &[_][]const u8{ config_path, "zba", "config.toml" }); | ||||
|     errdefer allocator.free(path); | ||||
| const CliError = error{ | ||||
|     InsufficientOptions, | ||||
|     UnneededOptions, | ||||
| }; | ||||
|  | ||||
|     // We try to create the file exclusively, meaning that we err out if the file already exists. | ||||
|     // All we care about is a file being there so we can just ignore that error in particular and | ||||
|     // continue down the happy pathj | ||||
|     std.fs.accessAbsolute(path, .{}) catch |e| { | ||||
|         if (e != error.FileNotFound) return e; | ||||
| // FIXME: Superfluous allocations? | ||||
| fn setupSavePath(alloc: std.mem.Allocator) !?[]const u8 { | ||||
|     const save_subpath = try std.fs.path.join(alloc, &[_][]const u8{ "zba", "save" }); | ||||
|     defer alloc.free(save_subpath); | ||||
|  | ||||
|         const config_file = std.fs.createFileAbsolute(path, .{}) catch |err| exitln("failed to create \"{s}\": {}", .{ path, err }); | ||||
|         defer config_file.close(); | ||||
|     const maybe_data_path = try known_folders.getPath(alloc, .data); | ||||
|     defer if (maybe_data_path) |path| alloc.free(path); | ||||
|  | ||||
|         try config_file.writeAll(@embedFile("../example.toml")); | ||||
|     }; | ||||
|     const save_path = if (maybe_data_path) |base| try std.fs.path.join(alloc, &[_][]const u8{ base, save_subpath }) else null; | ||||
|  | ||||
|     return path; | ||||
| } | ||||
|  | ||||
| fn ensureDataDirsExist(data_path: []const u8) !void { | ||||
|     var dir = try std.fs.openDirAbsolute(data_path, .{}); | ||||
|     defer dir.close(); | ||||
|  | ||||
|     // Will recursively create directories | ||||
|     try dir.makePath("zba" ++ std.fs.path.sep_str ++ "save"); | ||||
| } | ||||
|  | ||||
| fn ensureConfigDirExists(config_path: []const u8) !void { | ||||
|     var dir = try std.fs.openDirAbsolute(config_path, .{}); | ||||
|     defer dir.close(); | ||||
|  | ||||
|     try dir.makePath("zba"); | ||||
| } | ||||
|  | ||||
| fn romPath(result: *const clap.Result(clap.Help, ¶ms, clap.parsers.default)) []const u8 { | ||||
|     return switch (result.positionals.len) { | ||||
|         1 => result.positionals[0], | ||||
|         0 => exitln("ZBA requires a path to a GamePak ROM", .{}), | ||||
|         else => exitln("ZBA received too many positional arguments.", .{}), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| fn exitln(comptime format: []const u8, args: anytype) noreturn { | ||||
|     const stderr = std.io.getStdErr().writer(); | ||||
|     stderr.print(format, args) catch {}; // Just exit already... | ||||
|     stderr.writeByte('\n') catch {}; | ||||
|     std.os.exit(1); | ||||
|     if (save_path) |_| { | ||||
|         // If we've determined what our save path should be, ensure the prereq directories | ||||
|         // are present so that we can successfully write to the path when necessary | ||||
|         const maybe_data_dir = try known_folders.open(alloc, .data, .{}); | ||||
|         if (maybe_data_dir) |data_dir| try data_dir.makePath(save_subpath); | ||||
|     } | ||||
|  | ||||
|     return save_path; | ||||
| } | ||||
|   | ||||
							
								
								
									
										339
									
								
								src/platform.zig
									
									
									
									
									
								
							
							
						
						
									
										339
									
								
								src/platform.zig
									
									
									
									
									
								
							| @@ -1,339 +0,0 @@ | ||||
| const std = @import("std"); | ||||
| const SDL = @import("sdl2"); | ||||
| const gl = @import("gl"); | ||||
| const emu = @import("core/emu.zig"); | ||||
| const config = @import("config.zig"); | ||||
|  | ||||
| const Apu = @import("core/apu.zig").Apu; | ||||
| const Arm7tdmi = @import("core/cpu.zig").Arm7tdmi; | ||||
| const Scheduler = @import("core/scheduler.zig").Scheduler; | ||||
| const FpsTracker = @import("util.zig").FpsTracker; | ||||
|  | ||||
| const gba_width = @import("core/ppu.zig").width; | ||||
| const gba_height = @import("core/ppu.zig").height; | ||||
|  | ||||
| pub const sample_rate = 1 << 16; | ||||
| pub const sample_format = SDL.AUDIO_U16; | ||||
|  | ||||
| const default_title = "ZBA"; | ||||
|  | ||||
| pub const Gui = struct { | ||||
|     const Self = @This(); | ||||
|     const SDL_GLContext = *anyopaque; // SDL.SDL_GLContext is a ?*anyopaque | ||||
|     const log = std.log.scoped(.Gui); | ||||
|  | ||||
|     // zig fmt: off | ||||
|     const vertices: [32]f32 = [_]f32{ | ||||
|         // Positions        // Colours      // Texture Coords | ||||
|          1.0, -1.0, 0.0,    1.0, 0.0, 0.0,  1.0, 1.0, // Top Right | ||||
|          1.0,  1.0, 0.0,    0.0, 1.0, 0.0,  1.0, 0.0, // Bottom Right | ||||
|         -1.0,  1.0, 0.0,    0.0, 0.0, 1.0,  0.0, 0.0, // Bottom Left | ||||
|         -1.0, -1.0, 0.0,    1.0, 1.0, 0.0,  0.0, 1.0, // Top Left | ||||
|     }; | ||||
|  | ||||
|     const indices: [6]u32 = [_]u32{ | ||||
|         0, 1, 3, // First Triangle | ||||
|         1, 2, 3, // Second Triangle | ||||
|     }; | ||||
|     // zig fmt: on | ||||
|  | ||||
|     window: *SDL.SDL_Window, | ||||
|     ctx: SDL_GLContext, | ||||
|     title: []const u8, | ||||
|     audio: Audio, | ||||
|  | ||||
|     program_id: gl.GLuint, | ||||
|  | ||||
|     pub fn init(title: *const [12]u8, apu: *Apu, width: i32, height: i32) !Self { | ||||
|         if (SDL.SDL_Init(SDL.SDL_INIT_VIDEO | SDL.SDL_INIT_EVENTS | SDL.SDL_INIT_AUDIO) < 0) panic(); | ||||
|         if (SDL.SDL_GL_SetAttribute(SDL.SDL_GL_CONTEXT_PROFILE_MASK, SDL.SDL_GL_CONTEXT_PROFILE_CORE) < 0) panic(); | ||||
|         if (SDL.SDL_GL_SetAttribute(SDL.SDL_GL_CONTEXT_MAJOR_VERSION, 3) < 0) panic(); | ||||
|         if (SDL.SDL_GL_SetAttribute(SDL.SDL_GL_CONTEXT_MAJOR_VERSION, 3) < 0) panic(); | ||||
|  | ||||
|         const win_scale = @intCast(c_int, config.config().host.win_scale); | ||||
|  | ||||
|         const window = SDL.SDL_CreateWindow( | ||||
|             default_title, | ||||
|             SDL.SDL_WINDOWPOS_CENTERED, | ||||
|             SDL.SDL_WINDOWPOS_CENTERED, | ||||
|             @as(c_int, width * win_scale), | ||||
|             @as(c_int, height * win_scale), | ||||
|             SDL.SDL_WINDOW_OPENGL | SDL.SDL_WINDOW_SHOWN, | ||||
|         ) orelse panic(); | ||||
|  | ||||
|         const ctx = SDL.SDL_GL_CreateContext(window) orelse panic(); | ||||
|         if (SDL.SDL_GL_MakeCurrent(window, ctx) < 0) panic(); | ||||
|  | ||||
|         try gl.load(ctx, Self.glGetProcAddress); | ||||
|         if (SDL.SDL_GL_SetSwapInterval(@boolToInt(config.config().host.vsync)) < 0) panic(); | ||||
|  | ||||
|         const program_id = try compileShaders(); | ||||
|  | ||||
|         return Self{ | ||||
|             .window = window, | ||||
|             .title = std.mem.sliceTo(title, 0), | ||||
|             .ctx = ctx, | ||||
|             .program_id = program_id, | ||||
|             .audio = Audio.init(apu), | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn compileShaders() !gl.GLuint { | ||||
|         // TODO: Panic on Shader Compiler Failure + Error Message | ||||
|         const vert_shader = @embedFile("shader/pixelbuf.vert"); | ||||
|         const frag_shader = @embedFile("shader/pixelbuf.frag"); | ||||
|  | ||||
|         const vs = gl.createShader(gl.VERTEX_SHADER); | ||||
|         defer gl.deleteShader(vs); | ||||
|  | ||||
|         gl.shaderSource(vs, 1, &[_][*c]const u8{vert_shader}, 0); | ||||
|         gl.compileShader(vs); | ||||
|  | ||||
|         if (!shader.didCompile(vs)) return error.VertexCompileError; | ||||
|  | ||||
|         const fs = gl.createShader(gl.FRAGMENT_SHADER); | ||||
|         defer gl.deleteShader(fs); | ||||
|  | ||||
|         gl.shaderSource(fs, 1, &[_][*c]const u8{frag_shader}, 0); | ||||
|         gl.compileShader(fs); | ||||
|  | ||||
|         if (!shader.didCompile(fs)) return error.FragmentCompileError; | ||||
|  | ||||
|         const program = gl.createProgram(); | ||||
|         gl.attachShader(program, vs); | ||||
|         gl.attachShader(program, fs); | ||||
|         gl.linkProgram(program); | ||||
|  | ||||
|         return program; | ||||
|     } | ||||
|  | ||||
|     // Returns the VAO ID since it's used in run() | ||||
|     fn generateBuffers() struct { c_uint, c_uint, c_uint } { | ||||
|         var vao_id: c_uint = undefined; | ||||
|         var vbo_id: c_uint = undefined; | ||||
|         var ebo_id: c_uint = undefined; | ||||
|         gl.genVertexArrays(1, &vao_id); | ||||
|         gl.genBuffers(1, &vbo_id); | ||||
|         gl.genBuffers(1, &ebo_id); | ||||
|  | ||||
|         gl.bindVertexArray(vao_id); | ||||
|  | ||||
|         gl.bindBuffer(gl.ARRAY_BUFFER, vbo_id); | ||||
|         gl.bufferData(gl.ARRAY_BUFFER, @sizeOf(@TypeOf(vertices)), &vertices, gl.STATIC_DRAW); | ||||
|  | ||||
|         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, ebo_id); | ||||
|         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, @sizeOf(@TypeOf(indices)), &indices, gl.STATIC_DRAW); | ||||
|  | ||||
|         // Position | ||||
|         gl.vertexAttribPointer(0, 3, gl.FLOAT, gl.FALSE, 8 * @sizeOf(f32), @intToPtr(?*anyopaque, 0)); // lmao | ||||
|         gl.enableVertexAttribArray(0); | ||||
|         // Colour | ||||
|         gl.vertexAttribPointer(1, 3, gl.FLOAT, gl.FALSE, 8 * @sizeOf(f32), @intToPtr(?*anyopaque, (3 * @sizeOf(f32)))); | ||||
|         gl.enableVertexAttribArray(1); | ||||
|         // Texture Coord | ||||
|         gl.vertexAttribPointer(2, 2, gl.FLOAT, gl.FALSE, 8 * @sizeOf(f32), @intToPtr(?*anyopaque, (6 * @sizeOf(f32)))); | ||||
|         gl.enableVertexAttribArray(2); | ||||
|  | ||||
|         return .{ vao_id, vbo_id, ebo_id }; | ||||
|     } | ||||
|  | ||||
|     fn generateTexture(buf: []const u8) c_uint { | ||||
|         var tex_id: c_uint = undefined; | ||||
|         gl.genTextures(1, &tex_id); | ||||
|         gl.bindTexture(gl.TEXTURE_2D, tex_id); | ||||
|  | ||||
|         // gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); | ||||
|         // gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); | ||||
|  | ||||
|         gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST); | ||||
|         gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST); | ||||
|  | ||||
|         gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gba_width, gba_height, 0, gl.RGBA, gl.UNSIGNED_INT_8_8_8_8, buf.ptr); | ||||
|         // gl.generateMipmap(gl.TEXTURE_2D); // TODO: Remove? | ||||
|  | ||||
|         return tex_id; | ||||
|     } | ||||
|  | ||||
|     pub fn run(self: *Self, cpu: *Arm7tdmi, scheduler: *Scheduler) !void { | ||||
|         var quit = std.atomic.Atomic(bool).init(false); | ||||
|         var tracker = FpsTracker.init(); | ||||
|  | ||||
|         var buffer_ids = Self.generateBuffers(); | ||||
|         defer { | ||||
|             gl.deleteBuffers(1, &buffer_ids[2]); // EBO | ||||
|             gl.deleteBuffers(1, &buffer_ids[1]); // VBO | ||||
|             gl.deleteVertexArrays(1, &buffer_ids[0]); // VAO | ||||
|         } | ||||
|         const vao_id = buffer_ids[0]; | ||||
|  | ||||
|         const tex_id = Self.generateTexture(cpu.bus.ppu.framebuf.get(.Renderer)); | ||||
|         defer gl.deleteTextures(1, &tex_id); | ||||
|  | ||||
|         const thread = try std.Thread.spawn(.{}, emu.run, .{ &quit, scheduler, cpu, &tracker }); | ||||
|         defer thread.join(); | ||||
|  | ||||
|         var title_buf: [0x100]u8 = undefined; | ||||
|  | ||||
|         emu_loop: while (true) { | ||||
|             var event: SDL.SDL_Event = undefined; | ||||
|             while (SDL.SDL_PollEvent(&event) != 0) { | ||||
|                 switch (event.type) { | ||||
|                     SDL.SDL_QUIT => break :emu_loop, | ||||
|                     SDL.SDL_KEYDOWN => { | ||||
|                         const key_code = event.key.keysym.sym; | ||||
|                         var keyinput = cpu.bus.io.keyinput.load(.Monotonic); | ||||
|  | ||||
|                         switch (key_code) { | ||||
|                             SDL.SDLK_UP => keyinput.up.unset(), | ||||
|                             SDL.SDLK_DOWN => keyinput.down.unset(), | ||||
|                             SDL.SDLK_LEFT => keyinput.left.unset(), | ||||
|                             SDL.SDLK_RIGHT => keyinput.right.unset(), | ||||
|                             SDL.SDLK_x => keyinput.a.unset(), | ||||
|                             SDL.SDLK_z => keyinput.b.unset(), | ||||
|                             SDL.SDLK_a => keyinput.shoulder_l.unset(), | ||||
|                             SDL.SDLK_s => keyinput.shoulder_r.unset(), | ||||
|                             SDL.SDLK_RETURN => keyinput.start.unset(), | ||||
|                             SDL.SDLK_RSHIFT => keyinput.select.unset(), | ||||
|                             else => {}, | ||||
|                         } | ||||
|  | ||||
|                         cpu.bus.io.keyinput.store(keyinput.raw, .Monotonic); | ||||
|                     }, | ||||
|                     SDL.SDL_KEYUP => { | ||||
|                         const key_code = event.key.keysym.sym; | ||||
|                         var keyinput = cpu.bus.io.keyinput.load(.Monotonic); | ||||
|  | ||||
|                         switch (key_code) { | ||||
|                             SDL.SDLK_UP => keyinput.up.set(), | ||||
|                             SDL.SDLK_DOWN => keyinput.down.set(), | ||||
|                             SDL.SDLK_LEFT => keyinput.left.set(), | ||||
|                             SDL.SDLK_RIGHT => keyinput.right.set(), | ||||
|                             SDL.SDLK_x => keyinput.a.set(), | ||||
|                             SDL.SDLK_z => keyinput.b.set(), | ||||
|                             SDL.SDLK_a => keyinput.shoulder_l.set(), | ||||
|                             SDL.SDLK_s => keyinput.shoulder_r.set(), | ||||
|                             SDL.SDLK_RETURN => keyinput.start.set(), | ||||
|                             SDL.SDLK_RSHIFT => keyinput.select.set(), | ||||
|                             SDL.SDLK_i => { | ||||
|                                 comptime std.debug.assert(sample_format == SDL.AUDIO_U16); | ||||
|                                 log.err("Sample Count: {}", .{cpu.bus.apu.sample_queue.len() / 2}); | ||||
|                             }, | ||||
|                             // SDL.SDLK_j => log.err("Scheduler Capacity: {} | Scheduler Event Count: {}", .{ scheduler.queue.capacity(), scheduler.queue.count() }), | ||||
|                             SDL.SDLK_k => {}, | ||||
|                             else => {}, | ||||
|                         } | ||||
|  | ||||
|                         cpu.bus.io.keyinput.store(keyinput.raw, .Monotonic); | ||||
|                     }, | ||||
|                     else => {}, | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             // Emulator has an internal Double Buffer | ||||
|             const framebuf = cpu.bus.ppu.framebuf.get(.Renderer); | ||||
|             gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gba_width, gba_height, gl.RGBA, gl.UNSIGNED_INT_8_8_8_8, framebuf.ptr); | ||||
|  | ||||
|             gl.useProgram(self.program_id); | ||||
|             gl.bindVertexArray(vao_id); | ||||
|             gl.drawElements(gl.TRIANGLES, 6, gl.UNSIGNED_INT, null); | ||||
|             SDL.SDL_GL_SwapWindow(self.window); | ||||
|  | ||||
|             const dyn_title = std.fmt.bufPrintZ(&title_buf, "ZBA | {s} [Emu: {}fps] ", .{ self.title, tracker.value() }) catch unreachable; | ||||
|             SDL.SDL_SetWindowTitle(self.window, dyn_title.ptr); | ||||
|         } | ||||
|  | ||||
|         quit.store(true, .Monotonic); // Terminate Emulator Thread | ||||
|     } | ||||
|  | ||||
|     pub fn deinit(self: *Self) void { | ||||
|         self.audio.deinit(); | ||||
|         gl.deleteProgram(self.program_id); | ||||
|         SDL.SDL_GL_DeleteContext(self.ctx); | ||||
|         SDL.SDL_DestroyWindow(self.window); | ||||
|         SDL.SDL_Quit(); | ||||
|         self.* = undefined; | ||||
|     } | ||||
|  | ||||
|     fn glGetProcAddress(ctx: SDL.SDL_GLContext, proc: [:0]const u8) ?*anyopaque { | ||||
|         _ = ctx; | ||||
|         return SDL.SDL_GL_GetProcAddress(proc.ptr); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Audio = struct { | ||||
|     const Self = @This(); | ||||
|     const log = std.log.scoped(.PlatformAudio); | ||||
|  | ||||
|     device: SDL.SDL_AudioDeviceID, | ||||
|  | ||||
|     fn init(apu: *Apu) Self { | ||||
|         var have: SDL.SDL_AudioSpec = undefined; | ||||
|         var want: SDL.SDL_AudioSpec = std.mem.zeroes(SDL.SDL_AudioSpec); | ||||
|         want.freq = sample_rate; | ||||
|         want.format = sample_format; | ||||
|         want.channels = 2; | ||||
|         want.samples = 0x100; | ||||
|         want.callback = Self.callback; | ||||
|         want.userdata = apu; | ||||
|  | ||||
|         std.debug.assert(sample_format == SDL.AUDIO_U16); | ||||
|         log.info("Host Sample Rate: {}Hz, Host Format: SDL.AUDIO_U16", .{sample_rate}); | ||||
|  | ||||
|         const device = SDL.SDL_OpenAudioDevice(null, 0, &want, &have, 0); | ||||
|         if (device == 0) panic(); | ||||
|  | ||||
|         if (!config.config().host.mute) { | ||||
|             SDL.SDL_PauseAudioDevice(device, 0); // Unpause Audio | ||||
|             log.info("Unpaused Device", .{}); | ||||
|         } | ||||
|  | ||||
|         return .{ .device = device }; | ||||
|     } | ||||
|  | ||||
|     fn deinit(self: *Self) void { | ||||
|         SDL.SDL_CloseAudioDevice(self.device); | ||||
|         self.* = undefined; | ||||
|     } | ||||
|  | ||||
|     export fn callback(userdata: ?*anyopaque, stream: [*c]u8, len: c_int) void { | ||||
|         const T = *Apu; | ||||
|         const apu = @ptrCast(T, @alignCast(@alignOf(T), userdata)); | ||||
|  | ||||
|         comptime std.debug.assert(sample_format == SDL.AUDIO_U16); | ||||
|         const sample_buf = @ptrCast([*]u16, @alignCast(@alignOf(u16), stream))[0 .. @intCast(u32, len) / @sizeOf(u16)]; | ||||
|  | ||||
|         var previous: u16 = 0x8000; | ||||
|         for (sample_buf) |*sample| { | ||||
|             if (apu.sample_queue.pop()) |value| previous = value; | ||||
|  | ||||
|             sample.* = previous; | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const shader = struct { | ||||
|     const Kind = enum { vertex, fragment }; | ||||
|     const log = std.log.scoped(.Shader); | ||||
|  | ||||
|     fn didCompile(id: gl.GLuint) bool { | ||||
|         var success: gl.GLint = undefined; | ||||
|         gl.getShaderiv(id, gl.COMPILE_STATUS, &success); | ||||
|  | ||||
|         if (success == 0) err(id); | ||||
|  | ||||
|         return success == 1; | ||||
|     } | ||||
|  | ||||
|     fn err(id: gl.GLuint) void { | ||||
|         const buf_len = 512; | ||||
|         var error_msg: [buf_len]u8 = undefined; | ||||
|  | ||||
|         gl.getShaderInfoLog(id, buf_len, 0, &error_msg); | ||||
|         log.err("{s}", .{std.mem.sliceTo(&error_msg, 0)}); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| fn panic() noreturn { | ||||
|     const str = @as(?[*:0]const u8, SDL.SDL_GetError()) orelse "unknown error"; | ||||
|     @panic(std.mem.sliceTo(str, 0)); | ||||
| } | ||||
							
								
								
									
										780
									
								
								src/ppu.zig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										780
									
								
								src/ppu.zig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,780 @@ | ||||
| const std = @import("std"); | ||||
| const io = @import("bus/io.zig"); | ||||
|  | ||||
| const EventKind = @import("scheduler.zig").EventKind; | ||||
| const Scheduler = @import("scheduler.zig").Scheduler; | ||||
| const Arm7tdmi = @import("cpu.zig").Arm7tdmi; | ||||
|  | ||||
| const Bit = @import("bitfield").Bit; | ||||
| const Bitfield = @import("bitfield").Bitfield; | ||||
|  | ||||
| const Allocator = std.mem.Allocator; | ||||
| const log = std.log.scoped(.PPU); | ||||
| const pollBlankingDma = @import("bus/dma.zig").pollBlankingDma; | ||||
| const intToBytes = @import("util.zig").intToBytes; | ||||
|  | ||||
| /// This is used to generate byuu / Talurabi's Color Correction algorithm | ||||
| // const COLOUR_LUT = genColourLut(); | ||||
|  | ||||
| pub const width = 240; | ||||
| pub const height = 160; | ||||
| pub const framebuf_pitch = width * @sizeOf(u32); | ||||
|  | ||||
| pub const Ppu = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     // Registers | ||||
|  | ||||
|     bg: [4]Background, | ||||
|  | ||||
|     dispcnt: io.DisplayControl, | ||||
|     dispstat: io.DisplayStatus, | ||||
|     vcount: io.VCount, | ||||
|  | ||||
|     vram: Vram, | ||||
|     palette: Palette, | ||||
|     oam: Oam, | ||||
|     sched: *Scheduler, | ||||
|     framebuf: []u8, | ||||
|     alloc: Allocator, | ||||
|  | ||||
|     scanline_sprites: [128]?Sprite, | ||||
|     scanline_buf: [width]?u16, | ||||
|  | ||||
|     pub fn init(alloc: Allocator, sched: *Scheduler) !Self { | ||||
|         // Queue first Hblank | ||||
|         sched.push(.Draw, sched.tick + (240 * 4)); | ||||
|  | ||||
|         const framebuf = try alloc.alloc(u8, framebuf_pitch * height); | ||||
|         std.mem.set(u8, framebuf, 0); | ||||
|  | ||||
|         return Self{ | ||||
|             .vram = try Vram.init(alloc), | ||||
|             .palette = try Palette.init(alloc), | ||||
|             .oam = try Oam.init(alloc), | ||||
|             .sched = sched, | ||||
|             .framebuf = framebuf, | ||||
|             .alloc = alloc, | ||||
|  | ||||
|             // Registers | ||||
|             .bg = [_]Background{Background.init()} ** 4, | ||||
|             .dispcnt = .{ .raw = 0x0000 }, | ||||
|             .dispstat = .{ .raw = 0x0000 }, | ||||
|             .vcount = .{ .raw = 0x0000 }, | ||||
|  | ||||
|             .scanline_buf = [_]?u16{null} ** width, | ||||
|             .scanline_sprites = [_]?Sprite{null} ** 128, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn deinit(self: Self) void { | ||||
|         self.alloc.free(self.framebuf); | ||||
|         self.vram.deinit(); | ||||
|         self.palette.deinit(); | ||||
|         self.oam.deinit(); | ||||
|     } | ||||
|  | ||||
|     pub fn setBgOffsets(self: *Self, comptime n: u3, word: u32) void { | ||||
|         self.bg[n].hofs.raw = @truncate(u16, word); | ||||
|         self.bg[n].vofs.raw = @truncate(u16, word >> 16); | ||||
|     } | ||||
|  | ||||
|     pub fn setAdjCnts(self: *Self, comptime n: u3, word: u32) void { | ||||
|         self.bg[n].cnt.raw = @truncate(u16, word); | ||||
|         self.bg[n + 1].cnt.raw = @truncate(u16, word >> 16); | ||||
|     } | ||||
|  | ||||
|     /// Search OAM for Sprites that might be rendered on this scanline | ||||
|     fn fetchSprites(self: *Self) void { | ||||
|         const y = self.vcount.scanline.read(); | ||||
|  | ||||
|         var i: usize = 0; | ||||
|         search: while (i < self.oam.buf.len) : (i += 8) { | ||||
|             // Attributes in OAM are 6 bytes long, with 2 bytes of padding | ||||
|             // Grab Attributes from OAM | ||||
|             const attr0 = @bitCast(Attr0, self.oam.read(u16, i)); | ||||
|  | ||||
|             // Only consider enabled Sprites | ||||
|             if (!attr0.disabled.read()) { | ||||
|                 const attr1 = @bitCast(Attr1, self.oam.read(u16, i + 2)); | ||||
|  | ||||
|                 // When fetching sprites we only care about ones that could be rendered | ||||
|                 // on this scanline | ||||
|                 const iy = @bitCast(i8, y); | ||||
|  | ||||
|                 const start = attr0.y.read(); | ||||
|                 const istart = @bitCast(i8, start); | ||||
|  | ||||
|                 const end = start +% spriteDimensions(attr0.shape.read(), attr1.size.read())[1]; | ||||
|                 const iend = @bitCast(i8, end); | ||||
|  | ||||
|                 // Sprites are expected to be able to wraparound, we perform the same check | ||||
|                 // for unsigned and signed values so that we handle all valid sprite positions | ||||
|                 if ((start <= y and y < end) or (istart <= iy and iy < iend)) { | ||||
|                     for (self.scanline_sprites) |*maybe_sprite| { | ||||
|                         if (maybe_sprite.* == null) { | ||||
|                             maybe_sprite.* = Sprite.init(attr0, attr1, @bitCast(Attr2, self.oam.read(u16, i + 4))); | ||||
|                             continue :search; | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     log.err("Found more than 128 sprites in OAM Search", .{}); | ||||
|                     unreachable; // TODO: Is this truly unreachable? | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Draw all relevant sprites on a scanline | ||||
|     fn drawSprites(self: *Self, prio: u2) void { | ||||
|         const char_base = 0x4000 * 4; | ||||
|         const y = @bitCast(i8, self.vcount.scanline.read()); | ||||
|  | ||||
|         // Loop over every fetched sprite | ||||
|         sprite_loop: for (self.scanline_sprites) |maybe_sprites| { | ||||
|             if (maybe_sprites) |sprite| { | ||||
|                 // Move on to the next sprite If its of a different priority | ||||
|                 if (sprite.priority() != prio) continue :sprite_loop; | ||||
|  | ||||
|                 var i: u9 = 0; | ||||
|                 px_loop: while (i < sprite.width) : (i += 1) { | ||||
|                     const x = (sprite.x() +% i) % 240; | ||||
|                     const ix = @bitCast(i9, x); | ||||
|  | ||||
|                     // If We've already rendered a pixel here don't overwrite it | ||||
|                     if (self.scanline_buf[x] != null) continue :px_loop; | ||||
|  | ||||
|                     const start = sprite.x(); | ||||
|                     const istart = @bitCast(i9, start); | ||||
|  | ||||
|                     const end = start +% sprite.width; | ||||
|                     const iend = @bitCast(i9, end); | ||||
|  | ||||
|                     // By comparing with both signed and unsigned values we ensure that sprites | ||||
|                     // are displayed in all valid (AFAIK) configuration | ||||
|                     if ((start <= x and x < end) or (istart <= ix and ix < iend)) { | ||||
|                         self.drawSpritePixel(char_base, sprite, ix, y); | ||||
|                     } | ||||
|                 } | ||||
|             } else break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Draw a Pixel of a Sprite Tile | ||||
|     fn drawSpritePixel(self: *Self, char_base: u32, sprite: Sprite, x: i9, y: i8) void { | ||||
|         // FIXME: We branch on this condition quite a lot | ||||
|         const is_8bpp = sprite.is_8bpp(); | ||||
|  | ||||
|         // std.math.absInt is branchless | ||||
|         const x_diff = @bitCast(u9, std.math.absInt(x - @bitCast(i9, sprite.x())) catch unreachable); | ||||
|         const y_diff = @bitCast(u8, std.math.absInt(y -% @bitCast(i8, sprite.y())) catch unreachable); | ||||
|  | ||||
|         // Note that we flip the tile_pos not the (tile_pos % 8) like we do for | ||||
|         // Background Tiles. By doing this we mirror the entire sprite instead of | ||||
|         // just a specific tile (see how sprite.width and sprite.height are involved) | ||||
|         const tile_y = y_diff ^ if (sprite.v_flip()) (sprite.height - 1) else 0; | ||||
|         const tile_x = x_diff ^ if (sprite.h_flip()) (sprite.width - 1) else 0; | ||||
|  | ||||
|         // Like in the background Tiles are 8x8 groups of pixels in 8bpp or 4bpp formats | ||||
|         const tile_id = sprite.tile_id(); | ||||
|         const tile_row_offset: u32 = if (is_8bpp) 8 else 4; | ||||
|         const tile_len: u32 = if (is_8bpp) 0x40 else 0x20; | ||||
|  | ||||
|         const row = tile_y % 8; | ||||
|         const col = tile_x % 8; | ||||
|  | ||||
|         // When calcualting the inital address, the first entry is always 0x20 * tile_id, even if it is 8bpp | ||||
|         const tile_base = char_base + (0x20 * @as(u32, tile_id)) + (tile_row_offset * row) + if (is_8bpp) col else col / 2; | ||||
|  | ||||
|         // TODO: Understand more | ||||
|         var tile_offset = (tile_x / 8) * tile_len; | ||||
|         if (self.dispcnt.obj_mapping.read()) { | ||||
|             tile_offset += (tile_y / 8) * tile_len * (sprite.width / 8); // 1D Mapping | ||||
|         } else { | ||||
|             tile_offset += (tile_y / 8) * tile_len * 0x20; // 2D Mapping | ||||
|         } | ||||
|  | ||||
|         const tile = self.vram.buf[tile_base + tile_offset]; | ||||
|  | ||||
|         const pal_id: u16 = if (!is_8bpp) blk: { | ||||
|             const nybble_tile = if (col & 1 == 1) tile >> 4 else tile & 0xF; | ||||
|             if (nybble_tile == 0) break :blk 0; | ||||
|  | ||||
|             const pal_bank = @as(u8, sprite.pal_bank()) << 4; | ||||
|             break :blk pal_bank | nybble_tile; | ||||
|         } else tile; | ||||
|  | ||||
|         // Sprite Palette starts at 0x0500_0200 | ||||
|         if (pal_id != 0) self.scanline_buf[@bitCast(u9, x)] = self.palette.read(u16, 0x200 + pal_id * 2); | ||||
|     } | ||||
|  | ||||
|     fn drawBackround(self: *Self, comptime n: u3) void { | ||||
|         // A Tile in a charblock is a byte, while a Screen Entry is a halfword | ||||
|         const charblock_len: u32 = 0x4000; | ||||
|         const screenblock_len: u32 = 0x800; | ||||
|  | ||||
|         const cbb: u2 = self.bg[n].cnt.char_base.read(); // Char Block Base | ||||
|         const sbb: u5 = self.bg[n].cnt.screen_base.read(); // Screen Block Base | ||||
|         const is_8bpp: bool = self.bg[n].cnt.colour_mode.read(); // Colour Mode | ||||
|         const size: u2 = self.bg[n].cnt.size.read(); // Background Size | ||||
|  | ||||
|         // In 4bpp: 1 byte represents two pixels so the length is (8 x 8) / 2 | ||||
|         // In 8bpp: 1 byte represents one pixel so the length is 8 x 8 | ||||
|         const tile_len = if (is_8bpp) @as(u32, 0x40) else 0x20; | ||||
|         const tile_row_offset = if (is_8bpp) @as(u32, 0x8) else 0x4; | ||||
|  | ||||
|         // 0x0600_000 is implied because we can access VRAM without the Bus | ||||
|         const char_base: u32 = charblock_len * @as(u32, cbb); | ||||
|         const screen_base: u32 = screenblock_len * @as(u32, sbb); | ||||
|  | ||||
|         const vofs: u32 = self.bg[n].vofs.offset.read(); | ||||
|         const hofs: u32 = self.bg[n].hofs.offset.read(); | ||||
|  | ||||
|         const y = vofs + self.vcount.scanline.read(); | ||||
|  | ||||
|         var i: u32 = 0; | ||||
|         while (i < width) : (i += 1) { | ||||
|             // Exit early if a pixel is already here | ||||
|             if (self.scanline_buf[i] != null) continue; | ||||
|  | ||||
|             const x = hofs + i; | ||||
|  | ||||
|             // Grab the Screen Entry from VRAM | ||||
|             const entry_addr = screen_base + tilemapOffset(size, x, y); | ||||
|             const entry = @bitCast(ScreenEntry, self.vram.read(u16, entry_addr)); | ||||
|  | ||||
|             // Calculate the Address of the Tile in the designated Charblock | ||||
|             // We also take this opportunity to flip tiles if necessary | ||||
|             const tile_id: u32 = entry.tile_id.read(); | ||||
|             const row = if (entry.v_flip.read()) 7 - (y % 8) else y % 8; // Determine on which row in a tile we're on | ||||
|             const tile_addr = char_base + (tile_len * tile_id) + (tile_row_offset * row); | ||||
|  | ||||
|             // Calculate on which column in a tile we're on | ||||
|             // Similarly to when we calculated the row, if we're in 4bpp we want to account | ||||
|             // for 1 byte consisting of two pixels | ||||
|             const col = if (entry.h_flip.read()) 7 - (x % 8) else x % 8; | ||||
|             const tile = self.vram.buf[tile_addr + if (is_8bpp) col else col / 2]; | ||||
|  | ||||
|             // If we're in 8bpp, then the tile value is an index into the palette, | ||||
|             // If we're in 4bpp, we have to account for a pal bank value in the Screen entry | ||||
|             // and then we can index the palette | ||||
|             const pal_id: u16 = if (!is_8bpp) blk: { | ||||
|                 const nybble_tile = if (col & 1 == 1) tile >> 4 else tile & 0xF; | ||||
|                 if (nybble_tile == 0) break :blk 0; | ||||
|  | ||||
|                 const pal_bank = @as(u8, entry.pal_bank.read()) << 4; | ||||
|                 break :blk pal_bank | nybble_tile; | ||||
|             } else tile; | ||||
|  | ||||
|             if (pal_id != 0) self.scanline_buf[i] = self.palette.read(u16, pal_id * 2); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn drawScanline(self: *Self) void { | ||||
|         const bg_mode = self.dispcnt.bg_mode.read(); | ||||
|         const bg_enable = self.dispcnt.bg_enable.read(); | ||||
|         const obj_enable = self.dispcnt.obj_enable.read(); | ||||
|         const scanline = self.vcount.scanline.read(); | ||||
|  | ||||
|         switch (bg_mode) { | ||||
|             0x0 => { | ||||
|                 const fb_base = framebuf_pitch * @as(usize, scanline); | ||||
|                 if (obj_enable) self.fetchSprites(); | ||||
|  | ||||
|                 var i: usize = 0; | ||||
|                 while (i < 4) : (i += 1) { | ||||
|                     // Draw Sprites Here | ||||
|                     self.drawSprites(@truncate(u2, i)); | ||||
|                     if (i == self.bg[0].cnt.priority.read() and bg_enable & 1 == 1) self.drawBackround(0); | ||||
|                     if (i == self.bg[1].cnt.priority.read() and bg_enable >> 1 & 1 == 1) self.drawBackround(1); | ||||
|                     if (i == self.bg[2].cnt.priority.read() and bg_enable >> 2 & 1 == 1) self.drawBackround(2); | ||||
|                     if (i == self.bg[3].cnt.priority.read() and bg_enable >> 3 & 1 == 1) self.drawBackround(3); | ||||
|                 } | ||||
|  | ||||
|                 // Copy Drawn Scanline to Frame Buffer | ||||
|                 // If there are any nulls present in self.scanline_buf it means that no background drew a pixel there, so draw backdrop | ||||
|                 for (self.scanline_buf) |maybe_px, j| { | ||||
|                     const bgr555 = if (maybe_px) |px| px else self.palette.getBackdrop(); | ||||
|                     std.mem.copy(u8, self.framebuf[fb_base + j * @sizeOf(u32) ..][0..4], &intToBytes(u32, toRgba8888(bgr555))); | ||||
|                 } | ||||
|  | ||||
|                 // Reset Scanline Buffer | ||||
|                 std.mem.set(?u16, &self.scanline_buf, null); | ||||
|                 // Reset List of Sprites | ||||
|                 std.mem.set(?Sprite, &self.scanline_sprites, null); | ||||
|             }, | ||||
|             0x3 => { | ||||
|                 const fb_base = framebuf_pitch * @as(usize, scanline); | ||||
|                 const vram_base = width * @sizeOf(u16) * @as(usize, scanline); | ||||
|  | ||||
|                 var i: usize = 0; | ||||
|                 while (i < width) : (i += 1) { | ||||
|                     const bgr555 = self.vram.read(u16, vram_base + i * @sizeOf(u16)); | ||||
|                     std.mem.copy(u8, self.framebuf[fb_base + i * @sizeOf(u32) ..][0..4], &intToBytes(u32, toRgba8888(bgr555))); | ||||
|                 } | ||||
|             }, | ||||
|             0x4 => { | ||||
|                 const select = self.dispcnt.frame_select.read(); | ||||
|                 const fb_base = framebuf_pitch * @as(usize, scanline); | ||||
|                 const vram_base = width * @as(usize, scanline) + if (select) 0xA000 else @as(usize, 0); | ||||
|  | ||||
|                 // Render Current Scanline | ||||
|                 for (self.vram.buf[vram_base .. vram_base + width]) |byte, i| { | ||||
|                     const pal_id = @as(u16, byte) * @sizeOf(u16); | ||||
|                     const bgr555 = self.palette.read(u16, pal_id); | ||||
|  | ||||
|                     std.mem.copy(u8, self.framebuf[fb_base + i * @sizeOf(u32) ..][0..4], &intToBytes(u32, toRgba8888(bgr555))); | ||||
|                 } | ||||
|             }, | ||||
|             else => std.debug.panic("[PPU] TODO: Implement BG Mode {}", .{bg_mode}), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn tilemapOffset(size: u2, x: u32, y: u32) u32 { | ||||
|         // Current Row: (y % PIXEL_COUNT) / 8 | ||||
|         // Current COlumn: (x % PIXEL_COUNT) / 8 | ||||
|         // Length of 1 row of Screen Entries: 0x40 | ||||
|         // Length of 1 Screen Entry: 0x2 is the size of a screen entry | ||||
|         @setRuntimeSafety(false); | ||||
|  | ||||
|         return switch (size) { | ||||
|             0 => (x % 256 / 8) * 2 + (y % 256 / 8) * 0x40, // 256 x 256 | ||||
|             1 => blk: { | ||||
|                 // 512 x 256 | ||||
|                 const offset: u32 = if (x & 0x1FF > 0xFF) 0x800 else 0; | ||||
|                 break :blk offset + (x % 256 / 8) * 2 + (y % 256 / 8) * 0x40; | ||||
|             }, | ||||
|             2 => blk: { | ||||
|                 // 256 x 512 | ||||
|                 const offset: u32 = if (y & 0x1FF > 0xFF) 0x800 else 0; | ||||
|                 break :blk offset + (x % 256 / 8) * 2 + (y % 256 / 8) * 0x40; | ||||
|             }, | ||||
|             3 => blk: { | ||||
|                 // 512 x 512 | ||||
|                 const offset: u32 = if (x & 0x1FF > 0xFF) 0x800 else 0; | ||||
|                 const offset_2: u32 = if (y & 0x1FF > 0xFF) 0x800 else 0; | ||||
|                 break :blk offset + offset_2 + (x % 256 / 8) * 2 + (y % 512 / 8) * 0x40; | ||||
|             }, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn handleHDrawEnd(self: *Self, cpu: *Arm7tdmi, late: u64) void { | ||||
|         // Transitioning to a Hblank | ||||
|         if (self.dispstat.hblank_irq.read()) { | ||||
|             cpu.bus.io.irq.hblank.set(); | ||||
|             cpu.handleInterrupt(); | ||||
|         } | ||||
|  | ||||
|         // See if HBlank DMA is present and not enabled | ||||
|         pollBlankingDma(&cpu.bus, .HBlank); | ||||
|  | ||||
|         self.dispstat.hblank.set(); | ||||
|         self.sched.push(.HBlank, self.sched.now() + (68 * 4) - late); | ||||
|     } | ||||
|  | ||||
|     pub fn handleHBlankEnd(self: *Self, cpu: *Arm7tdmi, late: u64) void { | ||||
|         // The End of a Hblank (During Draw or Vblank) | ||||
|         const old_scanline = self.vcount.scanline.read(); | ||||
|         const scanline = (old_scanline + 1) % 228; | ||||
|  | ||||
|         self.vcount.scanline.write(scanline); | ||||
|         self.dispstat.hblank.unset(); | ||||
|  | ||||
|         // Perform Vc == VcT check | ||||
|         const coincidence = scanline == self.dispstat.vcount_trigger.read(); | ||||
|         self.dispstat.coincidence.write(coincidence); | ||||
|  | ||||
|         if (coincidence and self.dispstat.vcount_irq.read()) { | ||||
|             cpu.bus.io.irq.coincidence.set(); | ||||
|             cpu.handleInterrupt(); | ||||
|         } | ||||
|  | ||||
|         if (scanline < 160) { | ||||
|             // Transitioning to another Draw | ||||
|             self.sched.push(.Draw, self.sched.now() + (240 * 4) - late); | ||||
|         } else { | ||||
|             // Transitioning to a Vblank | ||||
|             if (scanline == 160) { | ||||
|                 self.dispstat.vblank.set(); | ||||
|  | ||||
|                 if (self.dispstat.vblank_irq.read()) { | ||||
|                     cpu.bus.io.irq.vblank.set(); | ||||
|                     cpu.handleInterrupt(); | ||||
|                 } | ||||
|  | ||||
|                 // See if Vblank DMA is present and not enabled | ||||
|                 pollBlankingDma(&cpu.bus, .VBlank); | ||||
|             } | ||||
|  | ||||
|             if (scanline == 227) self.dispstat.vblank.unset(); | ||||
|             self.sched.push(.VBlank, self.sched.now() + (240 * 4) - late); | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Palette = struct { | ||||
|     const palram_size = 0x400; | ||||
|     const Self = @This(); | ||||
|  | ||||
|     buf: []u8, | ||||
|     alloc: Allocator, | ||||
|  | ||||
|     fn init(alloc: Allocator) !Self { | ||||
|         const buf = try alloc.alloc(u8, palram_size); | ||||
|         std.mem.set(u8, buf, 0); | ||||
|  | ||||
|         return Self{ | ||||
|             .buf = buf, | ||||
|             .alloc = alloc, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn deinit(self: Self) void { | ||||
|         self.alloc.free(self.buf); | ||||
|     } | ||||
|  | ||||
|     pub fn read(self: *const Self, comptime T: type, address: usize) T { | ||||
|         const addr = address & 0x3FF; | ||||
|  | ||||
|         return switch (T) { | ||||
|             u32 => (@as(T, self.buf[addr + 3]) << 24) | (@as(T, self.buf[addr + 2]) << 16) | (@as(T, self.buf[addr + 1]) << 8) | (@as(T, self.buf[addr])), | ||||
|             u16 => (@as(T, self.buf[addr + 1]) << 8) | @as(T, self.buf[addr]), | ||||
|             u8 => self.buf[addr], | ||||
|             else => @compileError("PALRAM: Unsupported read width"), | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn write(self: *Self, comptime T: type, address: usize, value: T) void { | ||||
|         const addr = address & 0x3FF; | ||||
|  | ||||
|         switch (T) { | ||||
|             u32 => { | ||||
|                 self.buf[addr + 3] = @truncate(u8, value >> 24); | ||||
|                 self.buf[addr + 2] = @truncate(u8, value >> 16); | ||||
|                 self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|                 self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|             }, | ||||
|             u16 => { | ||||
|                 self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|                 self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|             }, | ||||
|             u8 => { | ||||
|                 const halfword: u16 = @as(u16, value) * 0x0101; | ||||
|                 const real_addr = addr & ~@as(u32, 1); // *was* 8-bit read so address won't be aligned | ||||
|  | ||||
|                 self.buf[real_addr + 1] = @truncate(u8, halfword >> 8); | ||||
|                 self.buf[real_addr + 0] = @truncate(u8, halfword >> 0); | ||||
|             }, | ||||
|             else => @compileError("PALRAM: Unsupported write width"), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn getBackdrop(self: *const Self) u16 { | ||||
|         return self.read(u16, 0); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Vram = struct { | ||||
|     const vram_size = 0x18000; | ||||
|     const Self = @This(); | ||||
|  | ||||
|     buf: []u8, | ||||
|     alloc: Allocator, | ||||
|  | ||||
|     fn init(alloc: Allocator) !Self { | ||||
|         const buf = try alloc.alloc(u8, vram_size); | ||||
|         std.mem.set(u8, buf, 0); | ||||
|  | ||||
|         return Self{ | ||||
|             .buf = buf, | ||||
|             .alloc = alloc, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn deinit(self: Self) void { | ||||
|         self.alloc.free(self.buf); | ||||
|     } | ||||
|  | ||||
|     pub fn read(self: *const Self, comptime T: type, address: usize) T { | ||||
|         const addr = Self.mirror(address); | ||||
|  | ||||
|         return switch (T) { | ||||
|             u32 => (@as(T, self.buf[addr + 3]) << 24) | (@as(T, self.buf[addr + 2]) << 16) | (@as(T, self.buf[addr + 1]) << 8) | (@as(T, self.buf[addr])), | ||||
|             u16 => (@as(T, self.buf[addr + 1]) << 8) | @as(T, self.buf[addr]), | ||||
|             u8 => self.buf[addr], | ||||
|             else => @compileError("VRAM: Unsupported read width"), | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn write(self: *Self, comptime T: type, dispcnt: io.DisplayControl, address: usize, value: T) void { | ||||
|         const mode: u3 = dispcnt.bg_mode.read(); | ||||
|         const addr = Self.mirror(address); | ||||
|  | ||||
|         switch (T) { | ||||
|             u32 => { | ||||
|                 self.buf[addr + 3] = @truncate(u8, value >> 24); | ||||
|                 self.buf[addr + 2] = @truncate(u8, value >> 16); | ||||
|                 self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|                 self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|             }, | ||||
|             u16 => { | ||||
|                 self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|                 self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|             }, | ||||
|             u8 => { | ||||
|                 // Ignore if write is in OBJ | ||||
|                 switch (mode) { | ||||
|                     0, 1, 2 => if (0x0601_0000 <= address and address < 0x0601_8000) return, | ||||
|                     else => if (0x0601_4000 <= address and address < 0x0601_8000) return, | ||||
|                 } | ||||
|  | ||||
|                 const halfword: u16 = @as(u16, value) * 0x0101; | ||||
|                 const real_addr = addr & ~@as(u32, 1); | ||||
|  | ||||
|                 self.buf[real_addr + 1] = @truncate(u8, halfword >> 8); | ||||
|                 self.buf[real_addr + 0] = @truncate(u8, halfword >> 0); | ||||
|             }, | ||||
|             else => @compileError("VRAM: Unsupported write width"), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn mirror(address: usize) usize { | ||||
|         const addr = address & 0x1FFFF; // repeated in steps of 128KiB | ||||
|         return if (addr >= 0x18000) addr & 0x7FFF else addr; // 64K + 32K + 32K (abcc) | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Oam = struct { | ||||
|     const oam_size = 0x400; | ||||
|     const Self = @This(); | ||||
|  | ||||
|     buf: []u8, | ||||
|     alloc: Allocator, | ||||
|  | ||||
|     fn init(alloc: Allocator) !Self { | ||||
|         const buf = try alloc.alloc(u8, oam_size); | ||||
|         std.mem.set(u8, buf, 0); | ||||
|  | ||||
|         return Self{ | ||||
|             .buf = buf, | ||||
|             .alloc = alloc, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     fn deinit(self: Self) void { | ||||
|         self.alloc.free(self.buf); | ||||
|     } | ||||
|  | ||||
|     pub fn read(self: *const Self, comptime T: type, address: usize) T { | ||||
|         const addr = address & 0x3FF; | ||||
|  | ||||
|         return switch (T) { | ||||
|             u32 => (@as(T, self.buf[addr + 3]) << 24) | (@as(T, self.buf[addr + 2]) << 16) | (@as(T, self.buf[addr + 1]) << 8) | (@as(T, self.buf[addr])), | ||||
|             u16 => (@as(T, self.buf[addr + 1]) << 8) | @as(T, self.buf[addr]), | ||||
|             u8 => self.buf[addr], | ||||
|             else => @compileError("OAM: Unsupported read width"), | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn write(self: *Self, comptime T: type, address: usize, value: T) void { | ||||
|         const addr = address & 0x3FF; | ||||
|  | ||||
|         switch (T) { | ||||
|             u32 => { | ||||
|                 self.buf[addr + 3] = @truncate(u8, value >> 24); | ||||
|                 self.buf[addr + 2] = @truncate(u8, value >> 16); | ||||
|                 self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|                 self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|             }, | ||||
|             u16 => { | ||||
|                 self.buf[addr + 1] = @truncate(u8, value >> 8); | ||||
|                 self.buf[addr + 0] = @truncate(u8, value >> 0); | ||||
|             }, | ||||
|             u8 => return, // 8-bit writes are explicitly ignored | ||||
|             else => @compileError("OAM: Unsupported write width"), | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Background = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     /// Read / Write | ||||
|     cnt: io.BackgroundControl, | ||||
|     /// Write Only | ||||
|     hofs: io.BackgroundOffset, | ||||
|     /// Write Only | ||||
|     vofs: io.BackgroundOffset, | ||||
|  | ||||
|     fn init() Self { | ||||
|         return .{ | ||||
|             .cnt = .{ .raw = 0x0000 }, | ||||
|             .hofs = .{ .raw = 0x0000 }, | ||||
|             .vofs = .{ .raw = 0x0000 }, | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const ScreenEntry = extern union { | ||||
|     tile_id: Bitfield(u16, 0, 10), | ||||
|     h_flip: Bit(u16, 10), | ||||
|     v_flip: Bit(u16, 11), | ||||
|     pal_bank: Bitfield(u16, 12, 4), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| const Sprite = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     attr0: Attr0, | ||||
|     attr1: Attr1, | ||||
|     attr2: Attr2, | ||||
|  | ||||
|     width: u8, | ||||
|     height: u8, | ||||
|  | ||||
|     fn init(attr0: Attr0, attr1: Attr1, attr2: Attr2) Self { | ||||
|         const d = spriteDimensions(attr0.shape.read(), attr1.size.read()); | ||||
|  | ||||
|         return .{ | ||||
|             .attr0 = attr0, | ||||
|             .attr1 = attr1, | ||||
|             .attr2 = attr2, | ||||
|             .width = d[0], | ||||
|             .height = d[1], | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     inline fn x(self: *const Self) u9 { | ||||
|         return self.attr1.x.read(); | ||||
|     } | ||||
|  | ||||
|     inline fn y(self: *const Self) u8 { | ||||
|         return self.attr0.y.read(); | ||||
|     } | ||||
|  | ||||
|     inline fn is_8bpp(self: *const Self) bool { | ||||
|         return self.attr0.is_8bpp.read(); | ||||
|     } | ||||
|  | ||||
|     inline fn shape(self: *const Self) u2 { | ||||
|         return self.attr0.shape.read(); | ||||
|     } | ||||
|  | ||||
|     inline fn size(self: *const Self) u2 { | ||||
|         return self.attr1.size.read(); | ||||
|     } | ||||
|  | ||||
|     inline fn tile_id(self: *const Self) u10 { | ||||
|         return self.attr2.tile_id.read(); | ||||
|     } | ||||
|  | ||||
|     inline fn pal_bank(self: *const Self) u4 { | ||||
|         return self.attr2.pal_bank.read(); | ||||
|     } | ||||
|  | ||||
|     inline fn h_flip(self: *const Self) bool { | ||||
|         return self.attr1.h_flip.read(); | ||||
|     } | ||||
|  | ||||
|     inline fn v_flip(self: *const Self) bool { | ||||
|         return self.attr1.v_flip.read(); | ||||
|     } | ||||
|  | ||||
|     inline fn priority(self: *const Self) u2 { | ||||
|         return self.attr2.rel_prio.read(); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const Attr0 = extern union { | ||||
|     y: Bitfield(u16, 0, 8), | ||||
|     rot_scaling: Bit(u16, 8), // This SBZ | ||||
|     disabled: Bit(u16, 9), | ||||
|     mode: Bitfield(u16, 10, 2), | ||||
|     mosaic: Bit(u16, 12), | ||||
|     is_8bpp: Bit(u16, 13), | ||||
|     shape: Bitfield(u16, 14, 2), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| const Attr1 = extern union { | ||||
|     x: Bitfield(u16, 0, 9), | ||||
|     h_flip: Bit(u16, 12), | ||||
|     v_flip: Bit(u16, 13), | ||||
|     size: Bitfield(u16, 14, 2), | ||||
|     raw: u16, | ||||
| }; | ||||
|  | ||||
| const Attr2 = extern union { | ||||
|     tile_id: Bitfield(u16, 0, 10), | ||||
|     rel_prio: Bitfield(u16, 10, 2), | ||||
|     pal_bank: Bitfield(u16, 12, 4), | ||||
| }; | ||||
|  | ||||
| fn spriteDimensions(shape: u2, size: u2) [2]u8 { | ||||
|     @setRuntimeSafety(false); | ||||
|  | ||||
|     return switch (shape) { | ||||
|         0b00 => switch (size) { | ||||
|             // Square | ||||
|             0b00 => [_]u8{ 8, 8 }, | ||||
|             0b01 => [_]u8{ 16, 16 }, | ||||
|             0b10 => [_]u8{ 32, 32 }, | ||||
|             0b11 => [_]u8{ 64, 64 }, | ||||
|         }, | ||||
|         0b01 => switch (size) { | ||||
|             0b00 => [_]u8{ 16, 8 }, | ||||
|             0b01 => [_]u8{ 32, 8 }, | ||||
|             0b10 => [_]u8{ 32, 16 }, | ||||
|             0b11 => [_]u8{ 64, 32 }, | ||||
|         }, | ||||
|         0b10 => switch (size) { | ||||
|             0b00 => [_]u8{ 8, 16 }, | ||||
|             0b01 => [_]u8{ 8, 32 }, | ||||
|             0b10 => [_]u8{ 16, 32 }, | ||||
|             0b11 => [_]u8{ 32, 64 }, | ||||
|         }, | ||||
|         else => std.debug.panic("{} is an invalid sprite shape", .{shape}), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| fn toRgba8888(bgr555: u16) u32 { | ||||
|     const b = @as(u32, bgr555 >> 10 & 0x1F); | ||||
|     const g = @as(u32, bgr555 >> 5 & 0x1F); | ||||
|     const r = @as(u32, bgr555 & 0x1F); | ||||
|  | ||||
|     return (r << 3 | r >> 2) << 24 | (g << 3 | g >> 2) << 16 | (b << 3 | b >> 2) << 8 | 0xFF; | ||||
| } | ||||
|  | ||||
| fn genColourLut() [0x8000]u32 { | ||||
|     return comptime { | ||||
|         @setEvalBranchQuota(std.math.maxInt(u32)); | ||||
|  | ||||
|         var lut: [0x8000]u32 = undefined; | ||||
|         for (lut) |*px, i| px.* = toRgba8888Talarubi(i); | ||||
|         return lut; | ||||
|     }; | ||||
| } | ||||
|  | ||||
| // FIXME: The implementation is incorrect and using it in the LUT crashes the compiler (OOM) | ||||
| /// Implementation courtesy of byuu and Talarubi at https://near.sh/articles/video/color-emulation | ||||
| fn toRgba8888Talarubi(bgr555: u16) u32 { | ||||
|     @setRuntimeSafety(false); | ||||
|  | ||||
|     const lcd_gamma: f64 = 4; | ||||
|     const out_gamma: f64 = 2.2; | ||||
|  | ||||
|     const b = @as(u32, bgr555 >> 10 & 0x1F); | ||||
|     const g = @as(u32, bgr555 >> 5 & 0x1F); | ||||
|     const r = @as(u32, bgr555 & 0x1F); | ||||
|  | ||||
|     const lb = std.math.pow(f64, @intToFloat(f64, b << 3 | b >> 2) / 31, lcd_gamma); | ||||
|     const lg = std.math.pow(f64, @intToFloat(f64, g << 3 | g >> 2) / 31, lcd_gamma); | ||||
|     const lr = std.math.pow(f64, @intToFloat(f64, r << 3 | r >> 2) / 31, lcd_gamma); | ||||
|  | ||||
|     const out_b = std.math.pow(f64, (220 * lb + 10 * lg + 50 * lr) / 255, 1 / out_gamma); | ||||
|     const out_g = std.math.pow(f64, (30 * lb + 230 * lg + 10 * lr) / 255, 1 / out_gamma); | ||||
|     const out_r = std.math.pow(f64, (0 * lb + 50 * lg + 255 * lr) / 255, 1 / out_gamma); | ||||
|  | ||||
|     return @floatToInt(u32, out_r) << 24 | @floatToInt(u32, out_g) << 16 | @floatToInt(u32, out_b) << 8 | 0xFF; | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| const std = @import("std"); | ||||
| 
 | ||||
| const Bus = @import("Bus.zig"); | ||||
| const Arm7tdmi = @import("cpu.zig").Arm7tdmi; | ||||
| const Clock = @import("bus/gpio.zig").Clock; | ||||
| 
 | ||||
| const Order = std.math.Order; | ||||
| const PriorityQueue = std.PriorityQueue; | ||||
| @@ -14,16 +14,15 @@ pub const Scheduler = struct { | ||||
|     tick: u64, | ||||
|     queue: PriorityQueue(Event, void, lessThan), | ||||
| 
 | ||||
|     pub fn init(allocator: Allocator) Self { | ||||
|         var sched = Self{ .tick = 0, .queue = PriorityQueue(Event, void, lessThan).init(allocator, {}) }; | ||||
|         sched.queue.add(.{ .kind = .HeatDeath, .tick = std.math.maxInt(u64) }) catch unreachable; | ||||
|     pub fn init(alloc: Allocator) Self { | ||||
|         var sched = Self{ .tick = 0, .queue = PriorityQueue(Event, void, lessThan).init(alloc, {}) }; | ||||
|         sched.push(.HeatDeath, std.math.maxInt(u64)); | ||||
| 
 | ||||
|         return sched; | ||||
|     } | ||||
| 
 | ||||
|     pub fn deinit(self: *Self) void { | ||||
|     pub fn deinit(self: Self) void { | ||||
|         self.queue.deinit(); | ||||
|         self.* = undefined; | ||||
|     } | ||||
| 
 | ||||
|     pub inline fn now(self: *const Self) u64 { | ||||
| @@ -36,38 +35,24 @@ pub const Scheduler = struct { | ||||
| 
 | ||||
|             switch (event.kind) { | ||||
|                 .HeatDeath => { | ||||
|                     log.err("u64 overflow. This *actually* should never happen.", .{}); | ||||
|                     log.err("A u64 overflowered. This *actually* should never happen.", .{}); | ||||
|                     unreachable; | ||||
|                 }, | ||||
|                 .Draw => { | ||||
|                     // The end of a VDraw | ||||
|                     cpu.bus.ppu.drawScanline(); | ||||
|                     cpu.bus.ppu.onHdrawEnd(cpu, late); | ||||
|                     cpu.bus.ppu.handleHDrawEnd(cpu, late); | ||||
|                 }, | ||||
|                 .TimerOverflow => |id| { | ||||
|                     switch (id) { | ||||
|                         inline 0...3 => |idx| cpu.bus.tim[idx].onTimerExpire(cpu, late), | ||||
|                         0 => cpu.bus.tim._0.handleOverflow(cpu, late), | ||||
|                         1 => cpu.bus.tim._1.handleOverflow(cpu, late), | ||||
|                         2 => cpu.bus.tim._2.handleOverflow(cpu, late), | ||||
|                         3 => cpu.bus.tim._3.handleOverflow(cpu, late), | ||||
|                     } | ||||
|                 }, | ||||
|                 .ApuChannel => |id| { | ||||
|                     switch (id) { | ||||
|                         0 => cpu.bus.apu.ch1.onToneSweepEvent(late), | ||||
|                         1 => cpu.bus.apu.ch2.onToneEvent(late), | ||||
|                         2 => cpu.bus.apu.ch3.onWaveEvent(late), | ||||
|                         3 => cpu.bus.apu.ch4.onNoiseEvent(late), | ||||
|                     } | ||||
|                 }, | ||||
|                 .RealTimeClock => { | ||||
|                     const device = &cpu.bus.pak.gpio.device; | ||||
|                     if (device.kind != .Rtc or device.ptr == null) return; | ||||
| 
 | ||||
|                     const clock = @ptrCast(*Clock, @alignCast(@alignOf(*Clock), device.ptr.?)); | ||||
|                     clock.onClockUpdate(late); | ||||
|                 }, | ||||
|                 .FrameSequencer => cpu.bus.apu.onSequencerTick(late), | ||||
|                 .SampleAudio => cpu.bus.apu.sampleAudio(late), | ||||
|                 .HBlank => cpu.bus.ppu.onHblankEnd(cpu, late), // The end of a HBlank | ||||
|                 .VBlank => cpu.bus.ppu.onHdrawEnd(cpu, late), // The end of a VBlank | ||||
|                 .HBlank => cpu.bus.ppu.handleHBlankEnd(cpu, late), // The end of a HBlank | ||||
|                 .VBlank => cpu.bus.ppu.handleHDrawEnd(cpu, late), // The end of a VBlank | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| @@ -92,15 +77,13 @@ pub const Scheduler = struct { | ||||
|     } | ||||
| 
 | ||||
|     pub fn push(self: *Self, kind: EventKind, end: u64) void { | ||||
|         self.queue.add(.{ .kind = kind, .tick = self.now() + end }) catch unreachable; | ||||
|         self.queue.add(.{ .kind = kind, .tick = end }) catch unreachable; | ||||
|     } | ||||
| 
 | ||||
|     pub inline fn nextTimestamp(self: *const Self) u64 { | ||||
|         @setRuntimeSafety(false); | ||||
|     pub fn nextTimestamp(self: *Self) u64 { | ||||
|         if (self.queue.peek()) |e| return e.tick; | ||||
| 
 | ||||
|         // Typically you'd use PriorityQueue.peek here, but there's always at least a HeatDeath | ||||
|         // event in the PQ so we can just do this instead. Should be faster in ReleaseSafe | ||||
|         return self.queue.items[0].tick; | ||||
|         unreachable; // There's always the HeatDeath event scheduled | ||||
|     } | ||||
| }; | ||||
| 
 | ||||
| @@ -119,8 +102,4 @@ pub const EventKind = union(enum) { | ||||
|     VBlank, | ||||
|     Draw, | ||||
|     TimerOverflow: u2, | ||||
|     SampleAudio, | ||||
|     FrameSequencer, | ||||
|     ApuChannel: u2, | ||||
|     RealTimeClock, | ||||
| }; | ||||
| @@ -1,25 +0,0 @@ | ||||
| #version 330 core | ||||
| out vec4 frag_color; | ||||
|  | ||||
| in vec3 color; | ||||
| in vec2 uv; | ||||
|  | ||||
| uniform sampler2D screen; | ||||
|  | ||||
| void main() { | ||||
| 	// https://near.sh/video/color-emulation | ||||
| 	// Thanks to Talarubi + Near for the Colour Correction | ||||
| 	// Thanks to fleur + mattrb for the Shader Impl | ||||
|  | ||||
| 	vec4 color = texture(screen, uv); | ||||
| 	color.rgb = pow(color.rgb, vec3(4.0)); // LCD Gamma | ||||
|    | ||||
| 	frag_color = vec4( | ||||
| 		pow(vec3( | ||||
| 		  	  0 * color.b +  50 * color.g + 255 * color.r, | ||||
| 	     	 30 * color.b + 230 * color.g +  10 * color.r, | ||||
| 			220 * color.b +  10 * color.g +  50 * color.r | ||||
| 		) / 255, vec3(1.0 / 2.2)), // Out Gamma | ||||
| 	1.0);  | ||||
| } | ||||
|  | ||||
| @@ -1,13 +0,0 @@ | ||||
| #version 330 core | ||||
| layout (location = 0) in vec3 pos; | ||||
| layout (location = 1) in vec3 in_color; | ||||
| layout (location = 2) in vec2 in_uv; | ||||
|  | ||||
| out vec3 color; | ||||
| out vec2 uv; | ||||
|  | ||||
| void main() { | ||||
| 	color = in_color; | ||||
| 	uv = in_uv; | ||||
| 	gl_Position = vec4(pos, 1.0); | ||||
| } | ||||
							
								
								
									
										380
									
								
								src/util.zig
									
									
									
									
									
								
							
							
						
						
									
										380
									
								
								src/util.zig
									
									
									
									
									
								
							| @@ -1,20 +1,11 @@ | ||||
| const std = @import("std"); | ||||
| const builtin = @import("builtin"); | ||||
| const config = @import("config.zig"); | ||||
|  | ||||
| const Log2Int = std.math.Log2Int; | ||||
| const Arm7tdmi = @import("core/cpu.zig").Arm7tdmi; | ||||
|  | ||||
| // Sign-Extend value of type `T` to type `U` | ||||
| pub fn sext(comptime T: type, comptime U: type, value: T) T { | ||||
|     // U must have less bits than T | ||||
|     comptime std.debug.assert(@typeInfo(U).Int.bits <= @typeInfo(T).Int.bits); | ||||
| pub inline fn sext(comptime bits: comptime_int, value: u32) u32 { | ||||
|     comptime std.debug.assert(bits <= 32); | ||||
|     const amount = 32 - bits; | ||||
|  | ||||
|     const iT = std.meta.Int(.signed, @typeInfo(T).Int.bits); | ||||
|     const ExtU = if (@typeInfo(U).Int.signedness == .unsigned) T else iT; | ||||
|     const shift_amt = @intCast(Log2Int(T), @typeInfo(T).Int.bits - @typeInfo(U).Int.bits); | ||||
|  | ||||
|     return @bitCast(T, @bitCast(iT, @as(ExtU, @truncate(U, value)) << shift_amt) >> shift_amt); | ||||
|     return @bitCast(u32, @bitCast(i32, value << amount) >> amount); | ||||
| } | ||||
|  | ||||
| /// See https://godbolt.org/z/W3en9Eche | ||||
| @@ -26,32 +17,30 @@ pub inline fn rotr(comptime T: type, x: T, r: anytype) T { | ||||
|     return x >> ar | x << (1 +% ~ar); | ||||
| } | ||||
|  | ||||
| pub const FpsTracker = struct { | ||||
| pub const FpsAverage = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     fps: u32, | ||||
|     count: std.atomic.Atomic(u32), | ||||
|     timer: std.time.Timer, | ||||
|     total: u64, | ||||
|     sample_count: u64, | ||||
|  | ||||
|     pub fn init() Self { | ||||
|         return .{ | ||||
|             .fps = 0, | ||||
|             .count = std.atomic.Atomic(u32).init(0), | ||||
|             .timer = std.time.Timer.start() catch unreachable, | ||||
|         }; | ||||
|         return .{ .total = 0, .sample_count = 1 }; | ||||
|     } | ||||
|  | ||||
|     pub fn tick(self: *Self) void { | ||||
|         _ = self.count.fetchAdd(1, .Monotonic); | ||||
|     pub fn add(self: *Self, sample: u64) void { | ||||
|         if (self.sample_count == 600) return self.reset(sample); | ||||
|  | ||||
|         self.total += sample; | ||||
|         self.sample_count += 1; | ||||
|     } | ||||
|  | ||||
|     pub fn value(self: *Self) u32 { | ||||
|         if (self.timer.read() >= std.time.ns_per_s) { | ||||
|             self.fps = self.count.swap(0, .Monotonic); | ||||
|             self.timer.reset(); | ||||
|         } | ||||
|     pub fn calc(self: *const Self) f64 { | ||||
|         return @intToFloat(f64, std.time.ns_per_s) / (@intToFloat(f64, self.total) / @intToFloat(f64, self.sample_count)); | ||||
|     } | ||||
|  | ||||
|         return self.fps; | ||||
|     fn reset(self: *Self, sample: u64) void { | ||||
|         self.total = sample; | ||||
|         self.sample_count = 1; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| @@ -66,17 +55,36 @@ pub fn intToBytes(comptime T: type, value: anytype) [@sizeOf(T)]u8 { | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| /// Creates a copy of a title with all Filesystem-invalid characters replaced | ||||
| /// The Title from the GBA Cartridge may be null padded to a maximum | ||||
| /// length of 12 bytes. | ||||
| /// | ||||
| /// This function returns a slice of everything just before the first | ||||
| /// `\0` | ||||
| pub fn correctTitle(title: [12]u8) []const u8 { | ||||
|     var len = title.len; | ||||
|     for (title) |char, i| { | ||||
|         if (char == 0) { | ||||
|             len = i; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return title[0..len]; | ||||
| } | ||||
|  | ||||
| /// Copies a Title and returns either an identical or similar | ||||
| /// array consisting of ASCII that won't make any file system angry | ||||
| /// | ||||
| /// e.g. POKEPIN R/S to POKEPIN R_S | ||||
| pub fn escape(title: [12]u8) [12]u8 { | ||||
|     var ret: [12]u8 = title; | ||||
| pub fn safeTitle(title: [12]u8) [12]u8 { | ||||
|     var result: [12]u8 = title; | ||||
|  | ||||
|     //TODO: Add more replacements | ||||
|     std.mem.replaceScalar(u8, &ret, '/', '_'); | ||||
|     std.mem.replaceScalar(u8, &ret, '\\', '_'); | ||||
|     for (result) |*char| { | ||||
|         if (char.* == '/' or char.* == '\\') char.* = '_'; | ||||
|         if (char.* == 0) break; | ||||
|     } | ||||
|  | ||||
|     return ret; | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| pub const FilePaths = struct { | ||||
| @@ -84,301 +92,3 @@ pub const FilePaths = struct { | ||||
|     bios: ?[]const u8, | ||||
|     save: ?[]const u8, | ||||
| }; | ||||
|  | ||||
| pub const io = struct { | ||||
|     pub const read = struct { | ||||
|         pub fn todo(comptime log: anytype, comptime format: []const u8, args: anytype) u8 { | ||||
|             log.debug(format, args); | ||||
|             return 0; | ||||
|         } | ||||
|  | ||||
|         pub fn undef(comptime T: type, comptime log: anytype, comptime format: []const u8, args: anytype) ?T { | ||||
|             @setCold(true); | ||||
|  | ||||
|             const unhandled_io = config.config().debug.unhandled_io; | ||||
|  | ||||
|             log.warn(format, args); | ||||
|             if (builtin.mode == .Debug and !unhandled_io) std.debug.panic("TODO: Implement I/O Register", .{}); | ||||
|  | ||||
|             return null; | ||||
|         } | ||||
|  | ||||
|         pub fn err(comptime T: type, comptime log: anytype, comptime format: []const u8, args: anytype) ?T { | ||||
|             @setCold(true); | ||||
|  | ||||
|             log.err(format, args); | ||||
|             return null; | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     pub const write = struct { | ||||
|         pub fn undef(log: anytype, comptime format: []const u8, args: anytype) void { | ||||
|             const unhandled_io = config.config().debug.unhandled_io; | ||||
|  | ||||
|             log.warn(format, args); | ||||
|             if (builtin.mode == .Debug and !unhandled_io) std.debug.panic("TODO: Implement I/O Register", .{}); | ||||
|         } | ||||
|     }; | ||||
| }; | ||||
|  | ||||
| pub const Logger = struct { | ||||
|     const Self = @This(); | ||||
|  | ||||
|     buf: std.io.BufferedWriter(4096 << 2, std.fs.File.Writer), | ||||
|  | ||||
|     pub fn init(file: std.fs.File) Self { | ||||
|         return .{ | ||||
|             .buf = .{ .unbuffered_writer = file.writer() }, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     pub fn print(self: *Self, comptime format: []const u8, args: anytype) !void { | ||||
|         try self.buf.writer().print(format, args); | ||||
|         try self.buf.flush(); // FIXME: On panics, whatever is in the buffer isn't written to file | ||||
|     } | ||||
|  | ||||
|     pub fn mgbaLog(self: *Self, cpu: *const Arm7tdmi, opcode: u32) void { | ||||
|         const fmt_base = "{X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} {X:0>8} cpsr: {X:0>8} | "; | ||||
|         const thumb_fmt = fmt_base ++ "{X:0>4}:\n"; | ||||
|         const arm_fmt = fmt_base ++ "{X:0>8}:\n"; | ||||
|  | ||||
|         if (cpu.cpsr.t.read()) { | ||||
|             if (opcode >> 11 == 0x1E) { | ||||
|                 // Instruction 1 of a BL Opcode, print in ARM mode | ||||
|                 const low = cpu.bus.dbgRead(u16, cpu.r[15] - 2); | ||||
|                 const bl_opcode = @as(u32, opcode) << 16 | low; | ||||
|  | ||||
|                 self.print(arm_fmt, Self.fmtArgs(cpu, bl_opcode)) catch @panic("failed to write to log file"); | ||||
|             } else { | ||||
|                 self.print(thumb_fmt, Self.fmtArgs(cpu, opcode)) catch @panic("failed to write to log file"); | ||||
|             } | ||||
|         } else { | ||||
|             self.print(arm_fmt, Self.fmtArgs(cpu, opcode)) catch @panic("failed to write to log file"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn fmtArgs(cpu: *const Arm7tdmi, opcode: u32) FmtArgTuple { | ||||
|         return .{ | ||||
|             cpu.r[0], | ||||
|             cpu.r[1], | ||||
|             cpu.r[2], | ||||
|             cpu.r[3], | ||||
|             cpu.r[4], | ||||
|             cpu.r[5], | ||||
|             cpu.r[6], | ||||
|             cpu.r[7], | ||||
|             cpu.r[8], | ||||
|             cpu.r[9], | ||||
|             cpu.r[10], | ||||
|             cpu.r[11], | ||||
|             cpu.r[12], | ||||
|             cpu.r[13], | ||||
|             cpu.r[14], | ||||
|             cpu.r[15] - if (cpu.cpsr.t.read()) 2 else @as(u32, 4), | ||||
|             cpu.cpsr.raw, | ||||
|             opcode, | ||||
|         }; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const FmtArgTuple = struct { u32, u32, u32, u32, u32, u32, u32, u32, u32, u32, u32, u32, u32, u32, u32, u32, u32, u32 }; | ||||
|  | ||||
| pub const audio = struct { | ||||
|     const _io = @import("core/bus/io.zig"); | ||||
|  | ||||
|     const ToneSweep = @import("core/apu/ToneSweep.zig"); | ||||
|     const Tone = @import("core/apu/Tone.zig"); | ||||
|     const Wave = @import("core/apu/Wave.zig"); | ||||
|     const Noise = @import("core/apu/Noise.zig"); | ||||
|  | ||||
|     pub const length = struct { | ||||
|         const FrameSequencer = @import("core/apu.zig").FrameSequencer; | ||||
|  | ||||
|         /// Update State of Ch1, Ch2 and Ch3 length timer | ||||
|         pub fn update(comptime T: type, self: *T, fs: *const FrameSequencer, nrx34: _io.Frequency) void { | ||||
|             comptime std.debug.assert(T == ToneSweep or T == Tone or T == Wave); | ||||
|  | ||||
|             // Write to NRx4 when FS's next step is not one that clocks the length counter | ||||
|             if (!fs.isLengthNext()) { | ||||
|                 // If length_enable was disabled but is now enabled and length timer is not 0 already, | ||||
|                 // decrement the length timer | ||||
|  | ||||
|                 if (!self.freq.length_enable.read() and nrx34.length_enable.read() and self.len_dev.timer != 0) { | ||||
|                     self.len_dev.timer -= 1; | ||||
|  | ||||
|                     // If Length Timer is now 0 and trigger is clear, disable the channel | ||||
|                     if (self.len_dev.timer == 0 and !nrx34.trigger.read()) self.enabled = false; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         pub const ch4 = struct { | ||||
|             /// update state of ch4 length timer | ||||
|             pub fn update(self: *Noise, fs: *const FrameSequencer, nr44: _io.NoiseControl) void { | ||||
|                 // Write to NRx4 when FS's next step is not one that clocks the length counter | ||||
|                 if (!fs.isLengthNext()) { | ||||
|                     // If length_enable was disabled but is now enabled and length timer is not 0 already, | ||||
|                     // decrement the length timer | ||||
|  | ||||
|                     if (!self.cnt.length_enable.read() and nr44.length_enable.read() and self.len_dev.timer != 0) { | ||||
|                         self.len_dev.timer -= 1; | ||||
|  | ||||
|                         // If Length Timer is now 0 and trigger is clear, disable the channel | ||||
|                         if (self.len_dev.timer == 0 and !nr44.trigger.read()) self.enabled = false; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
|     }; | ||||
| }; | ||||
|  | ||||
| /// Sets a quarter (8) of the bits of the u32 `left` to the value of u8 `right` | ||||
| pub inline fn setQuart(left: u32, addr: u8, right: u8) u32 { | ||||
|     const offset = @truncate(u2, addr); | ||||
|  | ||||
|     return switch (offset) { | ||||
|         0b00 => (left & 0xFFFF_FF00) | right, | ||||
|         0b01 => (left & 0xFFFF_00FF) | @as(u32, right) << 8, | ||||
|         0b10 => (left & 0xFF00_FFFF) | @as(u32, right) << 16, | ||||
|         0b11 => (left & 0x00FF_FFFF) | @as(u32, right) << 24, | ||||
|     }; | ||||
| } | ||||
|  | ||||
| /// Calculates the correct shift offset for an aligned/unaligned u8 read | ||||
| /// | ||||
| /// TODO: Support u16 reads of u32 values? | ||||
| pub inline fn getHalf(byte: u8) u4 { | ||||
|     return @truncate(u4, byte & 1) << 3; | ||||
| } | ||||
|  | ||||
| pub inline fn setHalf(comptime T: type, left: T, addr: u8, right: HalfInt(T)) T { | ||||
|     const offset = @truncate(u1, addr >> if (T == u32) 1 else 0); | ||||
|  | ||||
|     return switch (T) { | ||||
|         u32 => switch (offset) { | ||||
|             0b0 => (left & 0xFFFF_0000) | right, | ||||
|             0b1 => (left & 0x0000_FFFF) | @as(u32, right) << 16, | ||||
|         }, | ||||
|         u16 => switch (offset) { | ||||
|             0b0 => (left & 0xFF00) | right, | ||||
|             0b1 => (left & 0x00FF) | @as(u16, right) << 8, | ||||
|         }, | ||||
|         else => @compileError("unsupported type"), | ||||
|     }; | ||||
| } | ||||
|  | ||||
| /// The Integer type which corresponds to T with exactly half the amount of bits | ||||
| fn HalfInt(comptime T: type) type { | ||||
|     const type_info = @typeInfo(T); | ||||
|     comptime std.debug.assert(type_info == .Int); // Type must be an integer | ||||
|     comptime std.debug.assert(type_info.Int.bits % 2 == 0); // Type must have an even amount of bits | ||||
|  | ||||
|     return std.meta.Int(type_info.Int.signedness, type_info.Int.bits >> 1); | ||||
| } | ||||
|  | ||||
| const Mutex = std.Thread.Mutex; | ||||
|  | ||||
| pub fn RingBuffer(comptime T: type) type { | ||||
|     return struct { | ||||
|         const Self = @This(); | ||||
|         const Index = usize; | ||||
|         const max_capacity = (@as(Index, 1) << @typeInfo(Index).Int.bits - 1) - 1; // half the range of index type | ||||
|  | ||||
|         const log = std.log.scoped(.RingBuffer); | ||||
|  | ||||
|         read: Index, | ||||
|         write: Index, | ||||
|  | ||||
|         buf: []T, | ||||
|  | ||||
|         mutex: Mutex, | ||||
|  | ||||
|         const Error = error{buffer_full}; | ||||
|  | ||||
|         pub fn init(buf: []T) Self { | ||||
|             std.mem.set(T, buf, 0); | ||||
|  | ||||
|             std.debug.assert(std.math.isPowerOfTwo(buf.len)); // capacity must be a power of two | ||||
|             std.debug.assert(buf.len <= max_capacity); | ||||
|  | ||||
|             return .{ .read = 0, .write = 0, .buf = buf, .mutex = .{} }; | ||||
|         } | ||||
|  | ||||
|         pub fn push(self: *Self, left: T, right: T) Error!void { | ||||
|             self.mutex.lock(); | ||||
|             defer self.mutex.unlock(); | ||||
|  | ||||
|             try self._push(left); | ||||
|             self._push(right) catch |e| { | ||||
|                 self.write -= 1; // undo the previous write; | ||||
|                 return e; | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         pub fn pop(self: *Self) ?T { | ||||
|             self.mutex.lock(); | ||||
|             defer self.mutex.unlock(); | ||||
|  | ||||
|             return self._pop(); | ||||
|         } | ||||
|  | ||||
|         pub fn len(self: *Self) Index { | ||||
|             self.mutex.lock(); | ||||
|             defer self.mutex.unlock(); | ||||
|  | ||||
|             return self._len(); | ||||
|         } | ||||
|  | ||||
|         fn _push(self: *Self, value: T) Error!void { | ||||
|             if (self.isFull()) return error.buffer_full; | ||||
|             defer self.write += 1; | ||||
|  | ||||
|             self.buf[self.mask(self.write)] = value; | ||||
|         } | ||||
|  | ||||
|         fn _pop(self: *Self) ?T { | ||||
|             if (self.isEmpty()) return null; | ||||
|             defer self.read += 1; | ||||
|  | ||||
|             return self.buf[self.mask(self.read)]; | ||||
|         } | ||||
|  | ||||
|         fn _len(self: *const Self) Index { | ||||
|             return self.write - self.read; | ||||
|         } | ||||
|  | ||||
|         fn isFull(self: *const Self) bool { | ||||
|             return self._len() == self.buf.len; | ||||
|         } | ||||
|  | ||||
|         fn isEmpty(self: *const Self) bool { | ||||
|             return self.read == self.write; | ||||
|         } | ||||
|  | ||||
|         fn mask(self: *const Self, idx: Index) Index { | ||||
|             return idx & (self.buf.len - 1); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
|  | ||||
| test "RingBuffer" { | ||||
|     const Queue = RingBuffer(u8); | ||||
|  | ||||
|     var buf: [4]u8 = undefined; | ||||
|     var queue = Queue.init(&buf); | ||||
|  | ||||
|     try queue.push(1, 2); | ||||
|     try std.testing.expectEqual(@as(?u8, 1), queue.pop()); | ||||
|  | ||||
|     try queue.push(3, 4); | ||||
|     try std.testing.expectError(Queue.Error.buffer_full, queue.push(5, 6)); | ||||
|     try std.testing.expectEqual(@as(?u8, 2), queue.pop()); | ||||
|  | ||||
|     try queue.push(7, 8); | ||||
|  | ||||
|     try std.testing.expectEqual(@as(?u8, 3), queue.pop()); | ||||
|     try std.testing.expectEqual(@as(?u8, 4), queue.pop()); | ||||
|     try std.testing.expectEqual(@as(?u8, 7), queue.pop()); | ||||
|     try std.testing.expectEqual(@as(?u8, 8), queue.pop()); | ||||
|     try std.testing.expectEqual(@as(?u8, null), queue.pop()); | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user